diff --git a/.buildinfo b/.buildinfo new file mode 100644 index 00000000..1fa22492 --- /dev/null +++ b/.buildinfo @@ -0,0 +1,4 @@ +# Sphinx build info version 1 +# This file hashes the configuration used when building these files. When it is not found, a full rebuild will be done. +config: c7871e214e0a9c662e2f8c96b9b2ed87 +tags: 645f666f9bcd5a90fca523b33c5a78b7 diff --git a/.nojekyll b/.nojekyll new file mode 100644 index 00000000..e69de29b diff --git a/_modules/cogs/commands/analyze.html b/_modules/cogs/commands/analyze.html new file mode 100644 index 00000000..bd68ef49 --- /dev/null +++ b/_modules/cogs/commands/analyze.html @@ -0,0 +1,503 @@ + + + + + + + + + + cogs.commands.analyze — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.commands.analyze

+## > Imports
+# > Standard Library
+import datetime
+
+# > Discord imports
+import discord
+
+# > 3rd Party Dependencies
+from bs4 import BeautifulSoup
+from discord.commands import Option
+from discord.commands.context import ApplicationContext
+from discord.ext import commands
+
+# > Local dependencies
+from util.vars import get_json_data
+
+
+
+[docs] +class Analyze(commands.Cog): + """ + This class is used to handle the analyze command. + You can enable / disable this command in the config, under ["COMMANDS"]["ANALYZE"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + @commands.slash_command( + description="Request the current analysis for a stock ticker." + ) + async def analyze( + self, + ctx: ApplicationContext, + stock: Option(str, description="Stock ticker, e.g. AAPL.", required=True), + ) -> None: + """ + The analyze command is used to get the current analyst ratings for a stock ticker from benzinga.com. + + Parameters + ---------- + ctx : commands.Context + Discord context object. + stock : Option, optional + The ticker of a stock, e.g. AAPL + """ + + await ctx.response.defer(ephemeral=True) + + req = await get_json_data( + f"https://www.benzinga.com/quote/{stock}/analyst-ratings", text=True + ) + + soup = BeautifulSoup(req, "html.parser") + + tables = soup.find_all("tbody") + table = tables[1] + + headers = ["Buy", "Overweight", "Hold", "Underweight", "Sell"] + + data = [] + for row in table.find_all("td"): + data.append(row.text) + + e = discord.Embed( + title=f"{stock.upper()} Analysist Rating Summary ", + url=f"https://www.benzinga.com/quote/{stock}/analyst-ratings", + description="", + color=0x1F7FC1, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.set_footer( + text="\u200b", + icon_url="https://www.benzinga.com/next-assets/images/apple-touch-icon.png", + ) + + for i in range(len(headers)): + e.add_field(name=headers[i], value=data[i], inline=True) + + await ctx.respond(embed=e)
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Analyze(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/commands/earnings.html b/_modules/cogs/commands/earnings.html new file mode 100644 index 00000000..63e6436b --- /dev/null +++ b/_modules/cogs/commands/earnings.html @@ -0,0 +1,520 @@ + + + + + + + + + + cogs.commands.earnings — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.commands.earnings

+##> Imports
+from datetime import datetime
+
+import pandas as pd
+import pytz
+
+# > 3rd Party Dependencies
+import yfinance
+from discord.commands import Option
+from discord.commands.context import ApplicationContext
+from discord.ext import commands
+
+# Local dependencies
+from util.confirm_stock import confirm_stock
+
+
+
+[docs] +class Earnings(commands.Cog): + """ + This class is used to handle the earnings command. + You can enable / disable this command in the config, under ["COMMANDS"]["EARNINGS"]. + """ + + def __init__(self, bot: commands.Bot): + self.bot = bot + + @commands.slash_command( + name="earnings", description="Gets next earnings date for a given stock." + ) + async def earnings( + self, + ctx: ApplicationContext, + stock: Option(str, description="The requested stock.", required=True), + ): + """ + Gets next earnings date for a given stock. + For instance `/earnings AAPL` will return the next earnings date for Apple. + + Parameters + ---------- + ctx : commands.Context + Necessary Discord context object. + stock : str + The stock ticker to get the earnings date for. + + Raises + ------ + commands.UserInputError + If the provided stock ticker is not valid. + """ + + if input: + # Check if this stock exists + if not await confirm_stock(self.bot, ctx, stock): + return + + ticker = yfinance.Ticker(stock) + df = ticker.get_earnings_dates() + # Convert 'today' to a timezone-aware timestamp + tz = pytz.timezone("America/New_York") + today = pd.Timestamp(datetime.now(tz)) + + # Filter the DataFrame to include only future dates + future_dates = df[df.index > today] + + # Find the closest date + closest_date = future_dates.index.min() + + msg = f"The next earnings date for {stock.upper()} is <t:{closest_date.date()}:R>." + await ctx.respond(msg) + else: + raise commands.UserInputError() + +
+[docs] + @earnings.error + async def earnings_error(self, ctx: ApplicationContext, error: Exception): + """ + Catches the errors when using the `!earnings` command. + + Parameters + ---------- + ctx : commands.Context + Necessary Discord context object. + error : Exception + The exception that was raised when using the `!earnings` command. + """ + print(error) + if isinstance(error, commands.UserInputError): + await ctx.send( + f"{ctx.author.mention} You must specify a stock to request the next earnings of!" + ) + else: + await ctx.send( + f"{ctx.author.mention} An error has occurred. Please try again later." + )
+
+ + + +def setup(bot: commands.Bot): + bot.add_cog(Earnings(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/commands/help.html b/_modules/cogs/commands/help.html new file mode 100644 index 00000000..c5ca038e --- /dev/null +++ b/_modules/cogs/commands/help.html @@ -0,0 +1,552 @@ + + + + + + + + + + cogs.commands.help — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.commands.help

+##> Imports
+# > Discord dependencies
+import discord
+from discord.commands import Option
+from discord.commands.context import ApplicationContext
+from discord.ext import commands
+
+from util.disc_util import get_guild
+
+
+
+[docs] +class Help(commands.Cog): + """ + Custom help command. + """ + + def __init__(self, bot): + self.bot = bot + self.cmd_dict = {} + self.guild = None + + @commands.slash_command(description="Receive information about a command.") + async def help( + self, + ctx: ApplicationContext, + command: Option(str, description="Command to get help for.", required=False), + ): + """ + Receive information about a command or channel + Usage: `/help <command>` + List all commands available to you. If you want more information about a specific command, simply type that command after `/help`. + + + Parameters + ---------- + ctx : commands.Context + The context of the command. + command : Option, optional + A specific command, by default it will show all commands, required=False) + """ + await ctx.response.defer(ephemeral=True) + + if self.cmd_dict == {}: + self.get_cmd_dict() + + if not self.guild: + self.guild = get_guild(self.bot) + + # List all commands + if not command: + e = discord.Embed( + title="Available commands", + color=self.guild.self_role.color, + description="Use `/help <command>` to get more information about a command!", + ) + + cmd_mentions = [] + cmd_descs = [] + + for v in list(self.cmd_dict.values()): + cmd_mentions.append(v[0]) + cmd_descs.append(v[1]) + + e.add_field(name="Commands", value="\n".join(cmd_mentions), inline=True) + e.add_field(name="Description", value="\n".join(cmd_descs), inline=True) + + await ctx.respond(embed=e) + else: + command = command.lower() + + if command in self.cmd_dict.keys(): + e = discord.Embed( + title=f"The {command} command", + color=self.guild.self_role.color, + description="", + ) + + options = [] + for option in self.cmd_dict[command][2]: + options.append(f"**{option.name}**: {option.description}") + + e.add_field(name="Command", value=self.cmd_dict[command][0]) + e.add_field(name="Description", value=self.cmd_dict[command][1]) + e.add_field(name="Parameters", value="\n".join(options)) + await ctx.respond(embed=e) + else: + await ctx.respond(f"The {command} command was not found.") + +
+[docs] + def get_cmd_dict(self): + self.cmd_dict = {} + + # Iterate through all commands + for _, cog in self.bot.cogs.items(): + commands = cog.get_commands() + # https://docs.pycord.dev/en/stable/api.html?highlight=slashcommand#discord.SlashCommand + for command in commands: + # https://docs.pycord.dev/en/stable/api.html?highlight=slashcommand#slashcommandgroup + if type(command) == discord.SlashCommandGroup: + for subcommand in command.walk_commands(): + self.cmd_dict[f"{command.name} {subcommand.name}"] = [ + subcommand.mention, + subcommand.description, + subcommand.options, + ] + + elif type(command) == discord.SlashCommand: + self.cmd_dict[command.name] = [ + command.mention, + command.description, + command.options, + ]
+
+ + + # @help.error + # async def help_error(self, ctx, error): + # if isinstance(error, commands.UserInputError): + # await ctx.send( + # f"Too many arguments given. Correct usage of this command: `!help [command]`." + # ) + # elif isinstance(error, commands.CommandNotFound): + # e = discord.Embed( + # title="Help", + # color=0x00FFFF, + # description="This command could not be found... Try `!help` to list all available commands.", + # ) + # await ctx.send(embed=e) + # else: + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Help(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/commands/portfolio.html b/_modules/cogs/commands/portfolio.html new file mode 100644 index 00000000..29887339 --- /dev/null +++ b/_modules/cogs/commands/portfolio.html @@ -0,0 +1,691 @@ + + + + + + + + + + cogs.commands.portfolio — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.commands.portfolio

+# > 3rd Party Dependencies
+import traceback
+
+import ccxt
+
+# > Discord dependencies
+import discord
+import pandas as pd
+from discord import Interaction, SelectOption
+from discord.commands import Option, SlashCommandGroup
+from discord.commands.context import ApplicationContext
+from discord.ext import commands
+from discord.ui import Select, View
+
+# Local dependencies
+import util.vars
+from cogs.loops.assets import Assets
+from cogs.loops.trades import Trades
+from util.db import update_db
+
+
+
+[docs] +class Portfolio(commands.Cog): + """ + This class is used to handle the portfolio command. + You can enable / disable this command in the config, under ["COMMANDS"]["PORTFOLIO"]. + """ + + # Create a slash command group + portfolios = SlashCommandGroup("portfolio", description="Manage your portfolio.") + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + +
+[docs] + def update_portfolio_db(self, new_db): + # Set the new portfolio so other functions can access it + util.vars.portfolio_db = new_db + + # Write to SQL database + update_db(new_db, "portfolio")
+ + + @portfolios.command( + name="add", description="Add a cryptocurrency portfolio to the database." + ) + async def add( + self, + ctx: ApplicationContext, + exchange: Option( + str, + description="Provide the name of your crypto exchange.", + required=True, + ), + key: Option( + str, + description="Provide your API key.", + required=True, + ), + secret: Option( + str, + description="Provide your API secret.", + required=True, + ), + passphrase: Option( + str, + description="Provide your API passphrase (only used for Kucoin).", + required=False, + ), + ) -> None: + """ + Adds your portfolio to the database. + + Usage: + `/portfolio add <exchange> <key> <secret> (<passphrase>)` to add your portfolio to the database. + + Parameters + ---------- + ctx : commands.context.Context + The context of the command, for instance the user who used it. + input : tuple + The information specified after `!portfolio`. + """ + + # Check if the exchange is supported + if exchange.lower() not in ["binance", "kucoin"]: + raise commands.BadArgument() + + if exchange.lower() == "kucoin": + if not passphrase: + raise commands.UserInputError() + ccxt_exchange = ccxt.kucoin( + {"apiKey": key, "secret": secret, "password": passphrase} + ) + + elif exchange.lower() == "binance": + ccxt_exchange = ccxt.binance({"apiKey": key, "secret": secret}) + + # Check if the API keys are valid + status = ccxt_exchange.fetch_status() + if status["status"] != "ok": + await ctx.respond( + f"Your API keys are not valid! Please check your API keys and try again." + ) + return + + new_data = pd.DataFrame( + { + "id": ctx.author.id, + "user": ctx.author.name, + "exchange": exchange.lower(), + "key": key, + "secret": secret, + "passphrase": passphrase, + }, + index=[0], + ) + + # Check if new_data already exists in portfolio_db + if not util.vars.portfolio_db.empty: # ensure the DB isn't empty + # Check for duplicates based on a subset of columns that should be unique together + # Adjust the subset columns as per your data's unique constraints + duplicate_entries = util.vars.portfolio_db[ + (util.vars.portfolio_db["user"] == ctx.author.name) + & (util.vars.portfolio_db["exchange"] == exchange.lower()) + & (util.vars.portfolio_db["key"] == key) + & (util.vars.portfolio_db["secret"] == secret) + ] + + if not duplicate_entries.empty: + # Handle the case where a duplicate is found + await ctx.respond("This portfolio already exists in the database.") + return + + # Update the databse + util.vars.portfolio_db = pd.concat( + [util.vars.portfolio_db, new_data], ignore_index=True + ) + update_db(util.vars.portfolio_db, "portfolio") + + await ctx.respond( + "Succesfully added your portfolio to the database!\n⚠️ Please ensure that you set the API for read-only access ⚠️" + ) + + # Init Exchanges to start websockets + Trades(self.bot, new_data) + # Post the assets + Assets(self.bot, new_data) + + @portfolios.command( + name="remove", description="Remove a portfolio from the database." + ) + async def remove( + self, + ctx: ApplicationContext, + ) -> None: + """ + `/portfolio remove` to remove a specific portfolio from your list. + """ + + rows = util.vars.portfolio_db[util.vars.portfolio_db["id"] == ctx.author.id] + if not rows.empty: + options = [] + for i, (_, row) in enumerate(rows.iterrows()): + description = f"Exchange: {row['exchange']}" + options.append( + SelectOption( + label=f"Portfolio {i+1} - {row['exchange']}", + description=description, + value=str(i), + ) + ) + + view = PortfolioSelectView(ctx, util.vars.portfolio_db) + view.select_portfolio.options = options + await ctx.respond("Select the portfolio you want to remove:", view=view) + await view.wait() + else: + await ctx.respond("Your portfolio could not be found") + + @commands.dm_only() + @portfolios.command( + name="show", description="Show the portfolio(s) in the database." + ) + async def show( + self, + ctx: ApplicationContext, + ) -> None: + """ + `/portfolio show` to show your portfolio(s) in our database. + """ + rows = util.vars.portfolio_db[util.vars.portfolio_db["id"] == ctx.author.id] + if not rows.empty: + for _, row in rows.iterrows(): + response = f"Exchange: {row['exchange']} \nKey: {row['key']} \nSecret: {row['secret']}" + + if row["passphrase"]: + response += f"\nPassphrase: {row['passphrase']}" + await ctx.respond(response) + else: + await ctx.respond("Your portfolio could not be found") + +
+[docs] + @add.error + async def add_error(self, ctx: ApplicationContext, error: Exception) -> None: + # print(traceback.format_exc()) + if isinstance(error, commands.BadArgument): + await ctx.respond( + f"The exchange you specified is currently not supported! \nSupported exchanges: Kucoin, Binance" + ) + elif isinstance(error, commands.UserInputError): + await ctx.respond( + f"If using `/portfolio add` with Kucoin, you must specify a passphrase!" + ) + else: + print(error) + await ctx.respond(f"An error has occurred. Please try again later.")
+ + +
+[docs] + @remove.error + async def remove_error(self, ctx: ApplicationContext, error: Exception) -> None: + # print(traceback.format_exc()) + await ctx.respond(f"An error has occurred. Please try again later.")
+ + +
+[docs] + @show.error + async def show_error(self, ctx: ApplicationContext, error: Exception) -> None: + print(traceback.format_exc()) + if isinstance(error, commands.PrivateMessageOnly): + await ctx.respond( + "Please only use the `/portfolio` command in private messages for security reasons." + ) + else: + await ctx.respond(f"An error has occurred. Please try again later.")
+
+ + + +
+[docs] +class PortfolioSelectView(View): + def __init__(self, ctx, portfolio_db): + super().__init__() + self.ctx = ctx + self.portfolio_db = portfolio_db + +
+[docs] + @discord.ui.select(placeholder="Select the portfolio to remove") + async def select_portfolio(self, select: Select, interaction: Interaction): + if interaction.user != self.ctx.author: + return await interaction.response.send_message( + "You are not authorized to confirm this action.", ephemeral=True + ) + + index = int(select.values[0]) + self.portfolio_db.drop(self.portfolio_db.index[index], inplace=True) + await interaction.response.send_message( + "Successfully removed the selected portfolio from the database!", + ephemeral=True, + )
+
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Portfolio(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/commands/sentiment.html b/_modules/cogs/commands/sentiment.html new file mode 100644 index 00000000..2552c14c --- /dev/null +++ b/_modules/cogs/commands/sentiment.html @@ -0,0 +1,636 @@ + + + + + + + + + + cogs.commands.sentiment — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.commands.sentiment

+## > Imports
+# > Standard Library
+import datetime
+
+# > Discord imports
+import discord
+import nltk
+
+# > 3rd Party Dependencies
+import pandas as pd
+from discord.commands import Option
+from discord.commands.context import ApplicationContext
+from discord.ext import commands
+from nltk.sentiment.vader import SentimentIntensityAnalyzer
+
+from util.confirm_stock import confirm_stock
+
+# > Local dependencies
+from util.vars import get_json_data
+
+
+
+[docs] +class Sentiment(commands.Cog): + """ + This class is used to handle the sentiment command. + You can enable / disable this command in the config, under ["COMMANDS"]["SENTIMENT"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + @commands.slash_command( + description="Request the current sentiment for a stock ticker." + ) + async def sentiment( + self, + ctx: ApplicationContext, + stock: Option(str, description="Stock ticker, i.e. AAPL.", required=True), + ) -> None: + """ + This method is used to handle the sentiment command. + Usage: `!sentiment <stock ticker>` for instance `!sentiment AAPL`. + + Parameters + ---------- + ctx : commands.context.Context + The context of the command. + stock : str + The stock ticker, i.e. AAPL. Specified at the end of the command. + + Returns + ------- + None + """ + + await ctx.response.defer(ephemeral=True) + + # Check if this stock exists + if not await confirm_stock(self.bot, ctx, stock): + return + + news_df = await self.get_news(stock) + + # Get the total mean + total_mean = news_df["Sentiment"].mean() + + # The mean of the most recent 50 articles + fifty_mean = news_df["Sentiment"].head(50).mean() + + # The mean of the most recent 10 articles + ten_mean = news_df["Sentiment"].head(10).mean() + + e = discord.Embed( + title=f"Sentiment of Latest {stock.upper()} News", + url=f"https://finviz.com/quote.ashx?t={stock}", + description="", + color=0xFFFFFF, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + e.set_footer( + text="\u200b", + icon_url="https://pbs.twimg.com/profile_images/554978836488540160/rqhRqbgQ_400x400.png", + ) + + # Start by showing the means + e.add_field(name="Last 10 Mean", value=f"{ten_mean:.2f}", inline=True) + e.add_field(name="Last 50 Mean", value=f"{fifty_mean:.2f}", inline=True) + e.add_field(name="Total Mean", value=f"{total_mean:.2f}", inline=True) + + # Display the last 10 articles + last_5 = news_df.head(5) + + dates = last_5["Date"].dt.strftime("%d/%m/%y").tolist() + headlines = last_5["Headline"].astype(str).tolist() + sentiments = last_5["Sentiment"].astype(str).tolist() + + for i in range(5): + e.add_field( + name="Date" if i == 0 else "\u200b", value=dates[i], inline=True + ) + e.add_field( + name="Headline" if i == 0 else "\u200b", + value=headlines[i], + inline=True, + ) + e.add_field( + name="Sentiment" if i == 0 else "\u200b", + value=sentiments[i], + inline=True, + ) + + await ctx.respond(embed=e) + +
+[docs] + @sentiment.error + async def sentiment_error(self, ctx: ApplicationContext, error: Exception) -> None: + """ + Catches the errors when using the `!sentiment` command. + + Parameters + ---------- + ctx : commands.Context + The context of the command. + error : Exception + The exception that was raised when using the `!sentiment` command. + """ + print(error) + await ctx.respond(f"An error has occurred. Please try again later.")
+ + +
+[docs] + async def get_news(self, ticker: str) -> pd.DataFrame: + """ + Get the latest news for a given stock ticker. + + Parameters + ---------- + ticker : str + The stock ticker to get the news for. + + Returns + ------- + pd.DataFrame + The latest news for a given stock ticker. + """ + + html = await get_json_data( + url=f"https://finviz.com/quote.ashx?t={ticker}", + headers={ + "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.54 Safari/537.36" + }, + text=True, + ) + + print(html) + + # Get everything part of id='news-table' + html = html[html.find('id="news-table"') :] + html = html[: html.find("</table>")] + + # Split headlines by <tr> until </tr> + headlines = html.split("<tr>")[1:] + + text_only = [] + last_date = "" + dates = [] + sentiment = [] + + for headline in headlines: + date = headline[ + headline.find('style="white-space:nowrap">') + + len('style="white-space:nowrap">') : headline.find("&nbsp;") + ] + + if date.startswith('ht">'): + date = last_date + " " + date[len('ht">') :] + else: + last_date = date.split()[0] + + # Month-date-year hour:minute AM/pm + # For instance May-23-22 11:31PM + dates.append(datetime.datetime.strptime(date, "%b-%d-%y %I:%M%p")) + + text = headline[ + headline.find('class="tab-link-news">') + + len('class="tab-link-news">') : headline.find("</a>") + ].replace("&amp;", "&") + + url = headline[ + headline.find("href=") + + len("href=") + + 1 : headline.find('" target="_blank"') + ] + + text_only.append(f"[{text}]({url})") + + try: + analyzer = SentimentIntensityAnalyzer() + sentiment.append(analyzer.polarity_scores(text)["compound"]) + except LookupError: + # Download the NLTK packages + nltk.download("vader_lexicon") + + # Try again + analyzer = SentimentIntensityAnalyzer() + sentiment.append(analyzer.polarity_scores(text)["compound"]) + + return pd.DataFrame( + {"Date": dates, "Headline": text_only, "Sentiment": sentiment} + )
+
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Sentiment(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/commands/stock.html b/_modules/cogs/commands/stock.html new file mode 100644 index 00000000..8b97e988 --- /dev/null +++ b/_modules/cogs/commands/stock.html @@ -0,0 +1,732 @@ + + + + + + + + + + cogs.commands.stock — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.commands.stock

+# > 3rd Party Dependencies
+import pandas as pd
+import yfinance as yf
+from discord.commands import Option, SlashCommandGroup
+from discord.commands.context import ApplicationContext
+
+# Discord imports
+from discord.ext import commands
+
+# Local dependencies
+import util.vars
+from util.confirm_stock import confirm_stock
+from util.db import merge_and_update, update_db
+from util.disc_util import get_channel
+from util.trades_msg import trades_msg
+from util.vars import config
+
+
+
+[docs] +class Stock(commands.Cog): + """ + This class handles the `/stock` command. + You can enable / disable this command in the config, under ["COMMANDS"]["STOCK"]. + """ + + # Create a slash command group + stocks = SlashCommandGroup("stock", description="Add stocks to your portfolio.") + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + self.channel = None + +
+[docs] + def update_assets_db(self, new_db): + """ + Updates the assets database. + + Parameters + ---------- + new_db : pandas.DataFrame + The new database to be written to the assets database. + + Returns + ------- + None + """ + + # Set the new portfolio so other functions can access it + util.vars.assets_db = new_db + + # Write to SQL database + update_db(new_db, "assets")
+ + + @stocks.command(name="add", description="Add a stock to your portfolio.") + async def add( + self, + ctx: ApplicationContext, + ticker: Option( + str, description="The ticker of the stock e.g., AAPL", required=True + ), + buying_price: Option( + str, + description="The price of the stock when you bought it, e.g., 106.40", + required=True, + ), + amount: Option( + str, + description="The amount of stocks that you own at this price, e.g., 2", + required=True, + ), + ) -> None: + """ + Add stocks to your portfolio. + Usage: + `/stock add <ticker> <buying price> <amount>` to add a stock to your portfolio + + + Parameters + ---------- + ctx : commands.context.Context + The context of the command, such as the user who used it. + input : tuple + The keywords used following the `!stock` command. + + Returns + ------- + None + """ + if self.channel is None: + self.channel = await get_channel( + self.bot, config["LOOPS"]["TRADES"]["CHANNEL"] + ) + + await ctx.response.defer(ephemeral=True) + + # Make sure that the user is aware of this stock's existence + if not await confirm_stock(self.bot, ctx, ticker): + return + + try: + amount = float(amount) + buying_price = float(buying_price) + except Exception: + await ctx.respond("Please provide a valid buying price and/or amount.") + return + + try: + price = yf.Ticker(ticker).info["regularMarketPrice"] + except Exception: + price = 0 + + # Add ticker to database + new_data = pd.DataFrame( + [ + { + "asset": ticker.upper(), + "buying_price": buying_price, + "owned": amount, + "exchange": "stock", + "id": ctx.author.id, + "user": ctx.author.name, + } + ] + ) + + old_db = util.vars.assets_db + + # Check if the user has this asset already + owned_in_db = old_db.loc[ + (old_db["id"] == ctx.author.id) & (old_db["asset"] == ticker.upper()) + ] + + # If the user does not yet own this stock + if owned_in_db.empty: + util.vars.assets_db = merge_and_update(old_db, new_data, "assets") + else: + # Increase the amount if everything is the same + same_price = old_db.loc[ + (old_db["id"] == ctx.author.id) + & (old_db["asset"] == ticker.upper()) + & (old_db["buying_price"] == buying_price) + ] + + if not same_price.empty: + old_db.loc[ + (old_db["id"] == ctx.author.id) + & (old_db["asset"] == ticker.upper()), + "owned", + ] += amount + + else: + # Get the old buying price and average it with the new one + old_buying_price = owned_in_db["buying_price"].values[0] + old_amount_owned = owned_in_db["owned"].values[0] + + new_buying_price = ( + old_buying_price * old_amount_owned + buying_price * amount + ) / (old_amount_owned + amount) + + # Update the buying price and amount owned + old_db.loc[ + (old_db["id"] == ctx.author.id) + & (old_db["asset"] == ticker.upper()), + "buying_price", + ] = new_buying_price + + # Update the amount owned + old_db.loc[ + (old_db["id"] == ctx.author.id) + & (old_db["asset"] == ticker.upper()), + "owned", + ] += amount + + self.update_assets_db(old_db) + await ctx.respond("Succesfully added your stock to the database!") + + # Send message in trades channel + await trades_msg( + "stocks", + self.channel, + ctx.author, + ticker, + "buy", + "market", + buying_price, + amount, + round(price * amount, 2), + None, + ) + + @stocks.command( + name="remove", description="Remove a specific stock from your portfolio." + ) + async def remove( + self, + ctx: ApplicationContext, + ticker: Option( + str, description="The ticker of the stock e.g., AAPL", required=True + ), + amount: Option( + str, + description="The amount of stocks that you want to delete, e.g., 2", + required=False, + ), + ) -> None: + """ + Usage: + `!stock remove <ticker> (<amount>)` to remove a stock from your portfolio + """ + if self.channel is None: + self.channel = await get_channel( + self.bot, config["LOOPS"]["TRADES"]["CHANNEL"] + ) + + await ctx.response.defer(ephemeral=True) + + old_db = util.vars.assets_db + + if not amount: + row = old_db.index[ + (old_db["id"] == ctx.author.id) & (old_db["asset"] == ticker) + ] + + # Update database + if not row.empty: + amount = old_db.loc[row, "owned"].values[0] + self.update_assets_db(old_db.drop(index=row)) + await ctx.respond( + f"Succesfully removed all {ticker.upper()} from your owned stocks!" + ) + else: + await ctx.respond("You do not own this stock!") + return + + else: + try: + amount = float(amount) + except Exception: + await ctx.respond("Please provide a valid amount.") + return + + row = old_db.loc[ + (old_db["id"] == ctx.author.id) & (old_db["asset"] == ticker) + ] + + # Update database + if not row.empty: + # Check the amount owned + owned_now = row["owned"].tolist()[0] + # if it is equal to or greater than the amount to remove, remove all + if float(amount) >= owned_now: + self.update_assets_db(old_db.drop(index=row.index)) + await ctx.respond( + f"Succesfully removed all {ticker.upper()} from your owned stocks!" + ) + else: + old_db.loc[ + (old_db["id"] == ctx.author.id) + & (old_db["asset"] == ticker.upper()), + "owned", + ] -= float(amount) + self.update_assets_db(old_db) + await ctx.respond( + f"Succesfully removed {amount} {ticker.upper()} from your owned stocks!" + ) + else: + await ctx.respond("You do not own this stock!") + return + + try: + price = yf.Ticker(ticker).info["regularMarketPrice"] + except Exception: + price = 0 + + buying_price = row["buying_price"].tolist()[0] + + # Send message in trades channel + await trades_msg( + "stocks", + self.channel, + ctx.author, + ticker, + "sold", + "market", + price, + amount, + round(price * amount, 2), + buying_price, + ) + + @stocks.command(name="show", description="Show the stocks in your portfolio.") + async def show(self, ctx: ApplicationContext) -> None: + """ + Usage: + `!stock show` to show the stocks in your portfolio + """ + await ctx.response.defer(ephemeral=True) + db = util.vars.assets_db + rows = db.loc[(db["id"] == ctx.author.id) & (db["exchange"] == "stock")] + if not rows.empty: + for _, row in rows.iterrows(): + # Maybe send this an embed + await ctx.respond( + f"Stock: {row['asset'].upper()} \nAmount: {row['owned']}" + ) + else: + await ctx.respond("You do not have any stocks")
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Stock(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/listeners/on_member_join.html b/_modules/cogs/listeners/on_member_join.html new file mode 100644 index 00000000..ad30c772 --- /dev/null +++ b/_modules/cogs/listeners/on_member_join.html @@ -0,0 +1,445 @@ + + + + + + + + + + cogs.listeners.on_member_join — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.listeners.on_member_join

+##> Imports
+# > Discord dependencies
+from discord.ext import commands
+
+
+
+[docs] +class On_member_join(commands.Cog): + def __init__(self, bot): + self.bot = bot + +
+[docs] + @commands.Cog.listener() + async def on_member_join(self, member) -> None: + """Sends a private message to the member when they join the server""" + + await member.send( + """Welcome to the server! You can use `/help` to get a list of all commands available to you. +For more information about a specific command, use `/help <command>`. +Be sure to add your portfolio API read-only keys to your profile using `/portfolio`.""" + )
+
+ + + +def setup(bot): + bot.add_cog(On_member_join(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/listeners/on_raw_reaction_add.html b/_modules/cogs/listeners/on_raw_reaction_add.html new file mode 100644 index 00000000..a48f11d0 --- /dev/null +++ b/_modules/cogs/listeners/on_raw_reaction_add.html @@ -0,0 +1,612 @@ + + + + + + + + + + cogs.listeners.on_raw_reaction_add — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.listeners.on_raw_reaction_add

+##> Imports
+# > Standard libraries
+from csv import writer
+
+# > Discord dependencies
+import discord
+from discord.ext import commands
+
+# > Local dependencies
+from util.disc_util import get_channel, get_webhook
+from util.vars import config
+
+
+
+[docs] +class On_raw_reaction_add(commands.Cog): + """ + This class is used to handle the on_raw_reaction_add event. + You can enable / disable this command in the config, under ["LISTENERS"]["ON_RAW_REACTION_ADD"]. + """ + + def __init__(self, bot): + self.bot = bot + self.channel = None + +
+[docs] + @commands.Cog.listener() + async def on_raw_reaction_add( + self, reaction: discord.RawReactionActionEvent + ) -> None: + """ + This function is called when a reaction is added to a message. + + Parameters + ---------- + reaction : discord.RawReactionActionEvent + The information about the reaction that was added. + + Returns + ------- + None + """ + if self.channel is None: + self.channel = await get_channel( + self.bot, config["LISTENERS"]["ON_RAW_REACTION_ADD"]["CHANNEL"] + ) + + # Ignore private messages + if reaction.guild_id is None: + return + + try: + # Load necessary variables + channel = self.bot.get_channel(reaction.channel_id) + try: + message = discord.utils.get( + await channel.history(limit=100).flatten(), id=reaction.message_id + ) + except Exception as e: + print(f"Error getting channel.history for {channel}. Error:", e) + return + + if reaction.user_id != self.bot.user.id: + if ( + str(reaction.emoji) == "🐻" + or str(reaction.emoji) == "🐂" + or str(reaction.emoji) == "🦆" + ): + await self.classify_reaction(reaction, message) + elif str(reaction.emoji) == "💸": + await self.highlight(message, reaction.member) + elif str(reaction.emoji) == "❤️": + await self.send_dm(message, reaction.member) + + except commands.CommandError as e: + print(e)
+ + +
+[docs] + async def classify_reaction( + self, reaction: discord.RawReactionActionEvent, message: discord.Message + ) -> None: + """ + This function gets called if a reaction was used for classifying a tweet. + + Parameters + ---------- + reaction : discord.RawReactionActionEvent + The information about the reaction that was added. + message : discord.Message + The message that the reaction was added to. + + Returns + ------- + None + """ + + with open("data/sentiment_data.csv", "a", newline="") as file: + writer_object = writer(file) + if str(reaction.emoji) == "🐻": + writer_object.writerow( + [message.embeds[0].description.replace("\n", " "), -1] + ) + elif str(reaction.emoji) == "🐂": + writer_object.writerow( + [message.embeds[0].description.replace("\n", " "), 1] + ) + elif str(reaction.emoji) == "🦆": + writer_object.writerow( + [message.embeds[0].description.replace("\n", " "), 0] + )
+ + +
+[docs] + async def highlight(self, message: discord.Message, user: discord.User) -> None: + """ + This function gets called if a reaction was used for highlighting a tweet. + + Parameters + ---------- + message : discord.Message + The tweet that should be posted in the highlight channel. + user : discord.User + The user that added this reaction to the tweet. + + Returns + ------- + None + """ + + # Get the old embed + e = message.embeds[0] + + # Get the Discord name of the user + e.set_footer( + text=f"{e.footer.text} | Highlighted by {str(user).split('#')[0]}", + icon_url=e.footer.icon_url, + ) + + if len(message.embeds) > 1: + image_e = [e] + [ + discord.Embed(url=em.url).set_image(url=em.image.url) + for em in message.embeds[1:] + ] + + webhook = await get_webhook(self.channel) + + # Wait so we can use this message as reference + await webhook.send( + embeds=image_e, + username="FinTwit", + wait=True, + avatar_url=self.bot.user.avatar.url, + ) + + else: + await self.channel.send(embed=e)
+ + +
+[docs] + async def send_dm(self, message: discord.Message, user: discord.User) -> None: + """ + This function gets called if a reaction was used for sending a tweet via DM. + + Parameters + ---------- + message : discord.Message + The tweet that should be send to the DM of the user. + user : discord.User + The user that added this reaction to the tweet. + + Returns + ------- + None + """ + + # Check if the message has an embed + if message.embeds == []: + return + + # Get the old embed + e = message.embeds[0] + + # Send the embed to the user + await user.send(embed=e)
+
+ + + +def setup(bot): + bot.add_cog(On_raw_reaction_add(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/assets.html b/_modules/cogs/loops/assets.html new file mode 100644 index 00000000..ec52681d --- /dev/null +++ b/_modules/cogs/loops/assets.html @@ -0,0 +1,799 @@ + + + + + + + + + + cogs.loops.assets — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.assets

+## > Imports
+# > Standard libraries
+from __future__ import annotations
+
+import asyncio
+import datetime
+
+# > 3rd Party Dependencies
+import discord
+import numpy as np
+import pandas as pd
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+# > Local dependencies
+import util.vars
+from util.cg_data import get_coin_info
+from util.db import update_db
+from util.disc_util import get_channel, get_guild, get_user
+from util.exchange_data import get_data
+from util.formatting import format_change, format_embed_length
+from util.vars import config
+from util.yf_data import get_stock_info
+
+
+
+[docs] +class Assets(commands.Cog): + """ + The class is responsible for posting the assets of Discord users. + You can enabled / disable it in config under ["LOOPS"]["ASSETS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + self.assets.start() + +
+[docs] + async def usd_value(self, asset: str, exchange: str) -> tuple[float, float]: + """ + Get the USD value of an asset, based on the exchange. + + Parameters + ---------- + asset : str + The ticker of the asset, i.e. 'BTC'. + exchange : str + The exchange the asset is on, currently only 'binance' and 'kucoin' are supported. + + Returns + ------- + float + The worth of this asset in USD. + """ + + usd_val = change = None + + if exchange.lower() != "stock": + _, _, _, usd_val, change, _ = await get_coin_info(asset) + else: + _, _, _, usd_val, change, _ = await get_stock_info( + asset, do_format_change=False + ) + if isinstance(usd_val, list): + usd_val = usd_val[0] + if isinstance(change, list): + change = change[0] + + return usd_val, change
+ + + @loop(hours=1) + async def assets(self) -> None: + """ + Only do this function at startup and if a new portfolio has been added. + Checks the account balances of accounts saved in portfolio db, then updates the assets db. + + Parameters + ---------- + portfolio_db : pd.DataFrame + The portfolio db or the db for a new user. + + Returns + ------- + None + """ + assets_db_columns = { + "asset": str, + "buying_price": float, + "owned": float, + "exchange": str, + "id": np.int64, + "user": str, + "worth": float, + "price": float, + "change": float, + } + + if util.vars.portfolio_db.empty: + print("No portfolios in the database.") + return + + # Drop all crypto assets, so we can update them + if not util.vars.assets_db.empty: + crypto_rows = util.vars.assets_db.index[ + util.vars.assets_db["exchange"] != "stock" + ].tolist() + assets_db = util.vars.assets_db.drop(index=crypto_rows) + else: + # Create a new database + assets_db = pd.DataFrame(columns=list(assets_db_columns.keys())) + + # Get the assets of each user + for _, row in util.vars.portfolio_db.iterrows(): + # Add this data to the assets db + exch_data = await get_data(row) + assets_db = pd.concat([assets_db, exch_data], ignore_index=True) + + # Ensure that the db knows the right types + assets_db = assets_db.astype(assets_db_columns) + + # Update the assets db + update_db(assets_db, "assets") + util.vars.assets_db = assets_db + + # Post the assets + await self.post_assets() + +
+[docs] + async def update_prices_and_changes(self, new_df: pd.DataFrame) -> pd.DataFrame: + """ + Updates the prices and changes of the stock assets in the DataFrame. + """ + # Filter DataFrame to only include rows where exchange is "stock" + stock_df = new_df[new_df["exchange"] == "stock"] + + # Asynchronously get price and change for each asset + async def get_price_change(row): + price, change = await self.usd_value(row["asset"], row["exchange"]) + return { + "price": 0 if price is None else round(price, 2), + "change": 0 if change is None else change, + "worth": ( + 0 + if price in [None, np.nan] + else round(price * float(row["owned"]), 2) + ), + } + + # Using asyncio.gather to run all async operations concurrently + results = await asyncio.gather( + *(get_price_change(row) for _, row in stock_df.iterrows()) + ) + + # Update the DataFrame with the results + for i, (index, row) in enumerate(stock_df.iterrows()): + new_df.at[index, "price"] = results[i]["price"] + new_df.at[index, "change"] = results[i]["change"] + new_df.at[index, "worth"] = results[i]["worth"] + + return new_df
+ + +
+[docs] + async def format_exchange( + self, + exchange_df: pd.DataFrame, + exchange: str, + e: discord.Embed, + ) -> discord.Embed: + """ + Formats the embed used for updating user's assets. + + Parameters + ---------- + exchange_df : pd.DataFrame + The dataframe of assets owned by a user. + exchange : str + The exchange the assets are on, currently only 'binance' and 'kucoin' are supported. + e : discord.Embed + The embed to be formatted. + old_worth : str + The worth of the user's assets before the update. + old_assets : str + The assets of the user before the update. + + Returns + ------- + discord.Embed + The new embed. + """ + + # Necessary to prevent panda warnings + new_df = exchange_df.copy() + + # Add stock data to the DataFrame + stock_df = util.vars.assets_db[util.vars.assets_db["exchange"] == "stock"] + if not stock_df.empty: + new_df = await self.update_prices_and_changes(new_df) + + # Remove everything after % in change + # new_df["change"] = new_df["change"].str.split("%").str[0] + + # Set the types (again) + new_df = new_df.astype( + { + "asset": str, + "buying_price": float, + "owned": float, + "exchange": str, + "id": np.int64, + "user": str, + "worth": float, + "price": float, + "change": float, # Make sure this is not the formatted change + } + ) + + # Format the price change + new_df["change"] = new_df["change"].apply(lambda x: format_change(x)) + + # Format price and change + new_df["price_change"] = ( + "$" + + new_df["price"].astype(str) + + " (" + + new_df["change"].astype(str) + + ")" + ) + + # Fill NaN values of worth + new_df["worth"] = new_df["worth"].fillna(0) + + # Set buying price to float + new_df["buying_price"] = new_df["buying_price"].astype(float) + + # Add worth_change column + new_df["worth_change"] = "?" + + # Calculate the worth_change percentage only where buying_price is not 0 + mask = new_df["buying_price"] != 0 + new_df.loc[mask, "worth_change"] = round( + ((new_df["price"] - new_df["buying_price"]) / new_df["buying_price"] * 100), + 2, + ) + + # Apply format_change to worth_change + new_df.loc[mask, "worth_change"] = new_df.loc[mask, "worth_change"].apply( + lambda x: format_change(x) + ) + + # Sort by usd value + new_df = new_df.sort_values(by=["worth"], ascending=False) + + new_df["worth"] = ( + "$" + + new_df["worth"].astype(str) + + " (" + + new_df["worth_change"].astype(str) + + ")" + ) + + # Create the list of string values + assets = "\n".join(new_df["asset"].to_list()) + prices = "\n".join(new_df["price_change"].to_list()) + worth = "\n".join(new_df["worth"].to_list()) + + # Ensure that the length is not bigger than allowed + assets, prices, worth = format_embed_length([assets, prices, worth]) + + exchange_title = exchange + if exchange.lower() in util.vars.custom_emojis.keys(): + exchange_title = f"{exchange} {util.vars.custom_emojis[exchange.lower()]}" + + # These are the new fields added to the embed + e.add_field(name=exchange_title, value=assets, inline=True) + e.add_field(name="Price", value=prices, inline=True) + e.add_field(name="Worth", value=worth, inline=True) + + return e
+ + +
+[docs] + async def post_assets(self) -> None: + """ + Posts the assets of the users that added their portfolio. + + Returns + ------- + None + """ + + # Use the user name as channel + for id in util.vars.assets_db["id"].unique(): + # Get the assets of this user + user_assets = util.vars.assets_db.loc[util.vars.assets_db["id"] == id] + + # Only post if there are assets + if not user_assets.empty: + # Get the Discord objects + channel = await self.get_user_channel(user_assets["user"].values[0]) + disc_user = await self.get_user(user_assets) + + e = discord.Embed( + title="", + description="", + color=0x1DA1F2, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + if disc_user: + e.set_author( + name=disc_user.name + "'s Assets", + icon_url=disc_user.display_avatar.url, + ) + + # Finally, format the embed before posting it + for exchange in ["Binance", "KuCoin", "Stock"]: + exchange_df = user_assets.loc[ + user_assets["exchange"] == exchange.lower() + ] + + if not exchange_df.empty: + e = await self.format_exchange(exchange_df, exchange, e) + + await channel.purge(limit=1) + await channel.send(embed=e)
+ + +
+[docs] + async def get_user_channel(self, name: str) -> discord.TextChannel: + """ + Based on the username returns the user specific channel. + + Parameters + ---------- + name : str + The name of the Discord user. + + Returns + ------- + discord.TextChannel + The user specific channel. + """ + channel_name = config["LOOPS"]["ASSETS"]["CHANNEL_PREFIX"] + name.lower() + + # If this channel does not exist make it + channel = await get_channel(self.bot, channel_name) + if channel is None: + guild = get_guild(self.bot) + channel = await guild.create_text_channel( + channel_name, category=config["CATEGORIES"]["USERS"] + ) + print(f"Created channel {channel_name}") + + return channel
+ + +
+[docs] + async def get_user(self, assets): + id = assets["id"].values[0] + disc_user = self.bot.get_user(id) + + if disc_user is None: + try: + disc_user = await get_user(self.bot, id) + except Exception as e: + print(f"Could not get user with id: {id}.\n{assets} \nError:", e) + + return disc_user
+
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Assets(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/earnings_overview.html b/_modules/cogs/loops/earnings_overview.html new file mode 100644 index 00000000..a98a336d --- /dev/null +++ b/_modules/cogs/loops/earnings_overview.html @@ -0,0 +1,587 @@ + + + + + + + + + + cogs.loops.earnings_overview — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.earnings_overview

+import datetime
+
+# > Discord dependencies
+import discord
+
+# > 3rd party dependencies
+import pandas as pd
+
+# Local dependencies
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.disc_util import get_channel, get_tagged_users
+from util.vars import config, data_sources, get_json_data
+
+
+
+[docs] +class Earnings_Overview(commands.Cog): + """ + This class is responsible for sending weekly overview of upcoming earnings. + You can enable / disable this command in the config, under ["LOOPS"]["EARNINGS_OVERVIEW"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + self.channel = None + + self.earnings.start() + +
+[docs] + def earnings_embed(self, df: pd.DataFrame, date: str) -> tuple[str, discord.Embed]: + # Create lists of the important info + tickers = "\n".join(df["symbol"].to_list()) + time_type = "\n".join(df["time"].to_list()) + epsestimate = "\n".join(df["epsForecast"].replace("nan", "N/A").to_list()) + + # Make an embed with these tickers and their earnings date + estimation + e = discord.Embed( + title=f"Earnings for {date}", + url=f"https://finance.yahoo.com/calendar/earnings?day={date}", + description="", + color=data_sources["yahoo"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.add_field(name="Stock", value=tickers, inline=True) + e.add_field(name="Time", value=time_type, inline=True) + e.add_field(name="Estimate", value=epsestimate, inline=True) + + e.set_footer( + text="\u200b", + icon_url=data_sources["yahoo"]["icon"], + ) + + tags = get_tagged_users(df["symbol"].to_list()) + + return tags, e
+ + +
+[docs] + async def get_earnings_in_date_range( + self, start_date, end_date + ) -> list[pd.DataFrame]: + dfs = [] + for i in range((end_date - start_date).days + 1): + date = start_date + datetime.timedelta(days=i) + df = await self.get_earnings_for_date(date) + dfs.append(df) + + return dfs
+ + +
+[docs] + async def get_earnings_for_date(self, date: datetime.datetime) -> pd.DataFrame: + # Convert datetime to string YYYY-MM-DD + date = date.strftime("%Y-%m-%d") + url = f"https://api.nasdaq.com/api/calendar/earnings?date={date}" + # Add headers to avoid 403 error + headers = { + "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/125.0.0.0 Safari/537.36", + "Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7", + "Accept-Language": "en,nl-NL;q=0.9,nl;q=0.8,en-CA;q=0.7,ja;q=0.6", + "Accept-Encoding": "gzip, deflate, br, zstd", + "Sec-Ch-Ua": '"Google Chrome";v="125", "Chromium";v="125", "Not.A/Brand";v="24"', + } + json = await get_json_data(url, headers=headers) + # Automatically ordered from highest to lowest market cap + if "data" not in json: + return pd.DataFrame() + df = pd.DataFrame(json["data"]["rows"]) + if df.empty: + return df + # Replace time with emojis + emoji_dict = { + "time-after-hours": "🌙", + "time-pre-market": "🌞", + "time-not-supplied": "❓", + } + df["time"] = df["time"].replace(emoji_dict) + return df
+ + +
+[docs] + def date_check(self) -> bool: + """ + Check if today is a sunday and if it's 12 o'clock. + + Returns + ---------- + bool: + True if today is a sunday and the market is closed. + """ + + if ( + datetime.datetime.today().weekday() == 6 + and datetime.datetime.utcnow().hour == 12 + ): + return True + return False
+ + + @loop(hours=1) + async def earnings(self) -> None: + """ + Checks every hour if today is a sunday and if the market is closed. + If that is the case a overview will be posted with the upcoming earnings. + + Returns + ---------- + None + """ + if self.channel is None: + self.channel = await get_channel( + self.bot, config["LOOPS"]["EARNINGS_OVERVIEW"]["CHANNEL"] + ) + + # Send this message every sunday at 12:00 UTC + if self.date_check(): + end_date = datetime.datetime.now() + datetime.timedelta(days=6) + start_date = datetime.datetime.now() + datetime.timedelta(days=1) + earnings_dfs = await self.get_earnings_in_date_range( + start_date, + end_date, + ) + + for earnings_df, i in zip( + earnings_dfs, range((end_date - start_date).days + 1) + ): + date = start_date + datetime.timedelta(days=i) + date_string = date.strftime("%Y-%m-%d") + + if earnings_df.empty: + print(f"No earnings found for {date_string}") + continue + + # Only use the top 10 per dataframe + # Could change this in min. 1 billion USD market cap + + tags, e = self.earnings_embed(earnings_df.head(10), date_string) + await self.channel.send(content=tags, embed=e)
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Earnings_Overview(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/events.html b/_modules/cogs/loops/events.html new file mode 100644 index 00000000..01db5108 --- /dev/null +++ b/_modules/cogs/loops/events.html @@ -0,0 +1,773 @@ + + + + + + + + + + cogs.loops.events — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.events

+import datetime
+import re
+from io import StringIO
+
+# > Discord dependencies
+import discord
+
+# 3rd party imports
+import pandas as pd
+import pytz
+from bs4 import BeautifulSoup
+from discord.ext import commands
+from discord.ext.tasks import loop
+from lxml.html import fromstring
+
+from util.disc_util import get_channel
+
+# Local dependencies
+from util.vars import config, data_sources, get_json_data, post_json_data
+
+
+
+[docs] +class Events(commands.Cog): + """ + This class is responsible for sending weekly overview of upcoming events. + You can enable / disable this command in the config, under ["LOOPS"]["EVENTS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + if config["LOOPS"]["EVENTS"]["FOREX"]["ENABLED"]: + self.forex_channel = None + self.post_events.start() + + if config["LOOPS"]["EVENTS"]["CRYPTO"]["ENABLED"]: + self.crypto_channel = None + self.post_crypto_events.start() + +
+[docs] + async def get_events(self): + """ + Gets the economic calendar from Investing.com for the next week. + The data contains the most important information for the USA and EU. + + Forked from: https://github.com/alvarobartt/investpy/blob/master/investpy/news.py + """ + + headers = { + "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/105.0.0.0 Safari/537.36", + "X-Requested-With": "XMLHttpRequest", + } + + data = { + "country[]": [72, 5], # USA and EU + "importance[]": 3, # Highest importance, 3 stars + "timeZone": 8, + "timeFilter": "timeRemain", + "currentTab": "nextWeek", + "submitFilters": 1, + "limit_from": 0, + } + + url = "https://www.investing.com/economic-calendar/Service/getCalendarFilteredData" + + req = await post_json_data(url, headers=headers, data=data) + root = fromstring(req["data"]) + table = root.xpath(".//tr") + + results = [] + + for reversed_row in table[::-1]: + id_ = reversed_row.get("id") + if id_ is not None: + id_ = id_.replace("eventRowId_", "") + + for row in table: + id_ = row.get("id") + if id_ is None: + curr_timescope = int(row.xpath("td")[0].get("id").replace("theDay", "")) + curr_date = datetime.datetime.fromtimestamp( + curr_timescope, tz=pytz.timezone("GMT") + ).strftime("%d/%m/%Y") + else: + id_ = id_.replace("eventRowId_", "") + + time = zone = currency = event = actual = forecast = previous = None + + if row.get("id").__contains__("eventRowId_"): + for value in row.xpath("td"): + if value.get("class").__contains__("first left"): + time = value.text_content() + elif value.get("class").__contains__("flagCur"): + zone = value.xpath("span")[0].get("title").lower() + currency = value.text_content().strip() + elif value.get("class") == "left event": + event = value.text_content().strip() + elif value.get("id") == "eventActual_" + id_: + actual = value.text_content().strip() + elif value.get("id") == "eventForecast_" + id_: + forecast = value.text_content().strip() + elif value.get("id") == "eventPrevious_" + id_: + previous = value.text_content().strip() + + results.append( + { + "id": id_, + "date": curr_date, + "time": time, + "zone": zone, + "currency": None if currency == "" else currency, + "event": event, + "actual": None if actual == "" else actual, + "forecast": None if forecast == "" else forecast, + "previous": None if previous == "" else previous, + } + ) + + return pd.DataFrame(results)
+ + + @loop(hours=1) + async def post_events(self): + """ + Checks every hour if today is a friday and if the market is closed. + If that is the case a overview will be posted with the upcoming earnings. + + Returns + ---------- + None + """ + if self.forex_channel is None: + self.forex_channel = get_channel( + self.bot, + config["LOOPS"]["EVENTS"]["CHANNEL"], + config["CATEGORIES"]["FOREX"], + ) + + # Send this message every friday at 23:00 UTC + if datetime.datetime.today().weekday() == 4: + if datetime.datetime.utcnow().hour == 23: + df = await self.get_events() + + # If time == "All Day" convert it to 00:00 + df["time"] = df["time"].str.replace("All Day", "00:00") + + # Create datetime + df["datetime"] = pd.to_datetime( + df["date"] + " " + df["time"], + format="%d/%m/%Y %H:%M", + ) + + # Convert datetime to unix timestamp + df["timestamp"] = df["datetime"].astype("int64") // 10**9 + + # Convert timestamp to Discord timestamp using mode F + df["timestamp"] = df["timestamp"].apply(lambda x: f"<t:{int(x)}:d>") + + # Replace zone names with emojis + df["zone"] = df["zone"].replace( + {"euro zone": "🇪🇺", "united states": "🇺🇸"} + ) + time = "\n".join(df["timestamp"]) + + # Do this if both forecast and previous are not NaN + if ( + not df["forecast"].isnull().all() + and not df["previous"].isnull().all() + ): + df["forecast|previous"] = df["forecast"] + " | " + df["previous"] + for_prev = "\n".join(df["forecast|previous"].astype(str)) + for_prev_title = "Forecast | Previous" + + else: + for_prev_title = "Previous" + for_prev = "\n".join(df["previous"].astype(str)) + + df["info"] = df["zone"] + " " + df["event"] + info = "\n".join(df["info"]) + + # Make an embed with these tickers and their earnings date + estimation + e = discord.Embed( + title="Events Upcoming Week", + url="https://www.investing.com/economic-calendar/", + description="", + color=data_sources["investing"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.add_field(name="Date", value=time, inline=True) + e.add_field(name="Event", value=info, inline=True) + e.add_field(name=for_prev_title, value=for_prev, inline=True) + + e.set_footer( + text="\u200b", + icon_url=data_sources["investing"]["icon"], + ) + + await self.forex_channel.send(embed=e) + +
+[docs] + async def get_crypto_calendar(self) -> pd.DataFrame: + """ + Gets the economic calendar from CryptoCraft.com for the next week. + + Returns + ------- + pd.DataFrame + The formatted DataFrame containing the economic calendar. + """ + html = await get_json_data( + url="https://www.cryptocraft.com/calendar", + headers={ + "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4240.193 Safari/537.36" + }, + text=True, + ) + + soup = BeautifulSoup(html, "html.parser") + + # Get the first table + table = soup.find("table") + + impact_emoji = { + "yel": "🟨", + "ora": "🟧", + "red": "🟥", + "gra": "⬜", + } + + impacts = [] + for row in table.find_all("tr")[2:]: # Skip the header row + # Get the impact value from the span class including "impact" + impact = row.find("span", class_=lambda s: s and "impact" in s) + if impact: + impact = impact.get("class", [])[-1][-3:] + impacts.append(impact_emoji[impact]) + + # Convert the table to a string and read it into a DataFrame + df = pd.read_html(StringIO(str(table)))[0] + + # Drop the first row + df = df.iloc[1:] + + # Drop rows where the first and second column values are the same + df = df[df.iloc[:, 0] != df.iloc[:, 1]] + + # Convert MultiIndex columns to regular columns + df.columns = ["_".join(col).strip() for col in df.columns.values] + + # Rename second column to time and fifth column to event + df.rename( + columns={ + df.columns[0]: "date", + df.columns[1]: "time", + df.columns[4]: "event", + df.columns[6]: "actual", + df.columns[7]: "forecast", + df.columns[8]: "previous", + }, + inplace=True, + ) + + # Drop third and fourth column + df.drop(df.columns[[2, 3, 5, 9]], axis=1, inplace=True) + + # Remove rows where event is NaN + df = df[df["event"].notna()] + + # Reset index + df.reset_index(drop=True, inplace=True) + + # Add impact column + df["impact"] = impacts + + # Use ffill() for forward fill + df["time"] = df["time"].ffill() + + # Mask for entries where 'time' does not match common time patterns (only checks for absence of typical hour-minute time format) + mask_no_time_pattern = df["time"].str.contains( + r"^\D*$|day", flags=re.IGNORECASE, na=False + ) + # Mask for entries with specific time (i.e., typical time patterns are present) + mask_time_specific = ~mask_no_time_pattern + + # Convert 'All Day' entries by appending the current year and no specific time + df.loc[mask_no_time_pattern, "datetime"] = pd.to_datetime( + df.loc[mask_no_time_pattern, "date"] + + " " + + str(datetime.datetime.now().year), + format="%a %b %d %Y", + errors="coerce", + ) + + # Convert specific time entries by appending the current year and the specific time + df.loc[mask_time_specific, "datetime"] = pd.to_datetime( + df.loc[mask_time_specific, "date"] + + " " + + str(datetime.datetime.now().year) + + " " + + df.loc[mask_time_specific, "time"], + format="%a %b %d %Y %I:%M%p", + errors="coerce", + ) + + return df
+ + + @loop(hours=24) + async def post_crypto_events(self): + if self.crypto_channel is None: + self.crypto_channel = await get_channel( + self.bot, + config["LOOPS"]["EVENTS"]["CHANNEL"], + config["CATEGORIES"]["CRYPTO"], + ) + + df = await self.get_crypto_calendar() + + # Make an embed with these tickers and their earnings date + estimation + e = discord.Embed( + title="Upcoming Crypto Events", + url="https://www.cryptocraft.com/calendar", + description="", + color=data_sources["cryptocraft"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + # Convert datetime to unix timestamp + df["timestamp"] = df["datetime"].astype("int64") // 10**9 + + # Convert timestamp to Discord timestamp using mode F + df["timestamp"] = df["timestamp"].apply(lambda x: f"<t:{int(x)}:d>") + + date = "\n".join(df["timestamp"]) + event = "\n".join(df["event"]) + impact = "\n".join(df["impact"]) + + e.add_field(name="Date", value=date, inline=True) + e.add_field(name="Event", value=event, inline=True) + e.add_field(name="Impact", value=impact, inline=True) + + e.set_footer( + text="\u200b", + icon_url=data_sources["cryptocraft"]["icon"], + ) + + await self.crypto_channel.send(embed=e)
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Events(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/funding.html b/_modules/cogs/loops/funding.html new file mode 100644 index 00000000..a4e79543 --- /dev/null +++ b/_modules/cogs/loops/funding.html @@ -0,0 +1,548 @@ + + + + + + + + + + cogs.loops.funding — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.funding

+import datetime
+
+# > Discord dependencies
+import discord
+
+# > 3rd party dependencies
+import pandas as pd
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.disc_util import get_channel
+
+# Local dependencies
+from util.vars import config, data_sources, get_json_data
+
+
+
+[docs] +class Funding(commands.Cog): + """ + This class is used to handle the funding loop. + This can be enabled / disabled in the config, under ["LOOPS"]["FUNDING"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + self.channel = None + self.funding.start() + + @loop(hours=4) + async def funding(self) -> None: + """ + This function gets the data from the funding API and posts it in the funding channel. + + Returns + ------- + None + """ + if self.channel is None: + self.channel = await get_channel( + self.bot, config["LOOPS"]["FUNDING"]["CHANNEL"] + ) + + # Get the JSON data from the Binance API + binance_data = await get_json_data( + "https://fapi.binance.com/fapi/v1/premiumIndex" + ) + + # If the call did not work + if not binance_data: + print("Could not get funding data...") + return + + # Cast to dataframe + try: + df = pd.DataFrame(binance_data) + except Exception as e: + print(f"Could not cast to dataframe, error: {e}") + return + + # Keep only the USDT pairs + df = df[df["symbol"].str.contains("USDT")] + + # Remove USDT from the symbol + df["symbol"] = df["symbol"].str.replace("USDT", "") + + # Set it to numeric + df["lastFundingRate"] = df["lastFundingRate"].apply(pd.to_numeric) + + # Sort on lastFundingRate, lowest to highest + sorted = df.sort_values(by="lastFundingRate", ascending=True) + + # Multiply by 100 to get the funding rate in percent + sorted["lastFundingRate"] = sorted["lastFundingRate"] * 100 + + # Round to 4 decimal places + sorted["lastFundingRate"] = sorted["lastFundingRate"].round(4) + + # Convert them back to string + sorted = sorted.astype(str) + + # Add percentage to it + sorted["lastFundingRate"] = sorted["lastFundingRate"] + "%" + + # Post the top 15 lowest + lowest = sorted.head(15) + + e = discord.Embed( + title="Binance Top 15 Lowest Funding Rates", + url="", + description="", + color=data_sources["binance"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + # Get time to next funding, unix is in milliseconds + nextFundingTime = int(lowest["nextFundingTime"].tolist()[0]) // 1000 + nextFundingTime = datetime.datetime.fromtimestamp(nextFundingTime) + + # Get difference + timeToNextFunding = nextFundingTime - datetime.datetime.now() + + # Set datetime and icon + e.set_footer( + text=f"Next funding in {str(timeToNextFunding).split('.')[0]}", + icon_url=data_sources["binance"]["icon"], + ) + + lowest_tickers = "\n".join(lowest["symbol"].tolist()) + lowest_rates = "\n".join(lowest["lastFundingRate"].tolist()) + + e.add_field( + name="Coin", + value=lowest_tickers, + inline=True, + ) + + e.add_field( + name="Funding Rate", + value=lowest_rates, + inline=True, + ) + + # Post the embed in the channel + await self.channel.purge(limit=1) + await self.channel.send(embed=e)
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Funding(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/gainers.html b/_modules/cogs/loops/gainers.html new file mode 100644 index 00000000..28afa223 --- /dev/null +++ b/_modules/cogs/loops/gainers.html @@ -0,0 +1,581 @@ + + + + + + + + + + cogs.loops.gainers — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.gainers

+import pandas as pd
+
+# > 3rd party dependencies
+import yahoo_fin.stock_info as si
+
+# > Discord dependencies
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.afterhours import afterHours
+from util.disc_util import get_channel
+from util.formatting import format_embed
+
+# Local dependencies
+from util.vars import config, get_json_data
+
+
+
+[docs] +class Gainers(commands.Cog): + """ + This class contains the cog for posting the top crypto and stocks gainers. + It can be enabled / disabled in the config under ["LOOPS"]["GAINERS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + if config["LOOPS"]["GAINERS"]["STOCKS"]["ENABLED"]: + self.stocks_channel = None + self.stocks.start() + + if config["LOOPS"]["GAINERS"]["CRYPTO"]["ENABLED"]: + self.crypto_gainers_channel = None + + if config["LOOPS"]["LOSERS"]["CRYPTO"]["ENABLED"]: + self.crypto_losers_channel = None + + if ( + config["LOOPS"]["GAINERS"]["CRYPTO"]["ENABLED"] + or config["LOOPS"]["LOSERS"]["CRYPTO"]["ENABLED"] + ): + self.crypto.start() + + @loop(hours=1) + async def crypto(self) -> None: + """ + This function will check the gainers and losers on Binance, using USDT as the base currency. + To prevent too many calls the losers are also done in this section. + + Returns + ------- + None + """ + + binance_data = await get_json_data("https://api.binance.com/api/v3/ticker/24hr") + + # If the call did not work + if not binance_data: + return + + # Cast to dataframe + try: + df = pd.DataFrame(binance_data) + except Exception as e: + print(f"Could not cast to dataframe, error: {e}") + return + + # Keep only the USDT pairs + df = df[df["symbol"].str.contains("USDT")] + + # Remove USDT from the symbol + df["symbol"] = df["symbol"].str.replace("USDT", "") + + df[["priceChangePercent", "weightedAvgPrice", "volume"]] = df[ + ["priceChangePercent", "weightedAvgPrice", "volume"] + ].apply(pd.to_numeric) + + # Sort on priceChangePercent + sorted = df.sort_values(by="priceChangePercent", ascending=False) + + sorted.rename( + columns={ + "symbol": "Symbol", + "priceChangePercent": "% Change", + "weightedAvgPrice": "Price", + "volume": "Volume", + }, + inplace=True, + ) + + # Add website to symbol + sorted["Symbol"] = ( + "[" + + sorted["Symbol"] + + "](https://www.binance.com/en/price/" + + sorted["Symbol"] + + ")" + ) + + # Post the top 10 highest + gainers = sorted.head(10) + + # Post the top 10 lowest + losers = sorted.tail(10) + losers = losers.iloc[::-1] + + # Format the embed + e_gainers = await format_embed(gainers, "Gainers", "binance") + e_losers = await format_embed(losers, "Losers", "binance") + + # Post the embed in the channel + if config["LOOPS"]["GAINERS"]["CRYPTO"]["ENABLED"]: + if self.crypto_gainers_channel is None: + self.crypto_gainers_channel = await get_channel( + self.bot, + config["LOOPS"]["GAINERS"]["CHANNEL"], + config["CATEGORIES"]["CRYPTO"], + ) + await self.crypto_gainers_channel.purge(limit=1) + await self.crypto_gainers_channel.send(embed=e_gainers) + + if config["LOOPS"]["LOSERS"]["CRYPTO"]["ENABLED"]: + if self.crypto_losers_channel is None: + self.crypto_losers_channel = await get_channel( + self.bot, + config["LOOPS"]["LOSERS"]["CHANNEL"], + config["CATEGORIES"]["CRYPTO"], + ) + await self.crypto_losers_channel.purge(limit=1) + await self.crypto_losers_channel.send(embed=e_losers) + + @loop(hours=1) + async def stocks(self) -> None: + """ + This function uses the yahoo_fin.stock_info module to get the gainers for todays stocks. + + Returns + ------- + None + """ + if self.stocks_channel is None: + self.stocks_channel = await get_channel( + self.bot, + config["LOOPS"]["GAINERS"]["CHANNEL"], + config["CATEGORIES"]["STOCKS"], + ) + + # Dont send if the market is closed + if afterHours(): + return + + try: + e = await format_embed(si.get_day_gainers().head(10), "Gainers", "yahoo") + await self.stocks_channel.purge(limit=1) + await self.stocks_channel.send(embed=e) + except Exception as e: + print("Error posting stocks gainers: ", e)
+ + # print(traceback.format_exc()) + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Gainers(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/ideas.html b/_modules/cogs/loops/ideas.html new file mode 100644 index 00000000..743e4a8b --- /dev/null +++ b/_modules/cogs/loops/ideas.html @@ -0,0 +1,846 @@ + + + + + + + + + + cogs.loops.ideas — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.ideas

+import datetime
+import json
+import re
+
+import discord
+import pandas as pd
+from bs4 import BeautifulSoup
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+import util.vars
+from util.db import update_db
+from util.disc_util import get_channel, get_tagged_users
+from util.vars import config, data_sources, get_json_data
+
+
+
+[docs] +class TradingView_Ideas(commands.Cog): + """ + This class contains the cog for posting the latest Trading View ideas. + It can be enabled / disabled in the config under ["LOOPS"]["TV_IDEAS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + if config["LOOPS"]["IDEAS"]["CRYPTO"]["ENABLED"]: + self.crypto_channel = None + self.crypto_ideas.start() + + if config["LOOPS"]["IDEAS"]["STOCKS"]["ENABLED"]: + self.stocks_channel = None + self.stock_ideas.start() + + if config["LOOPS"]["IDEAS"]["FOREX"]["ENABLED"]: + self.forex_channel = None + self.forex_ideas.start() + +
+[docs] + def add_id_to_db(self, id: str) -> None: + """ + Adds the given id to the database. + """ + + util.vars.ideas_ids = pd.concat( + [ + util.vars.ideas_ids, + pd.DataFrame( + [ + { + "id": id, + "timestamp": datetime.datetime.now(), + } + ] + ), + ], + ignore_index=True, + )
+ + +
+[docs] + async def send_embed(self, df: pd.DataFrame, type: str) -> None: + """ + Creates an embed based on the given DataFrame and type. + Then sends this embed in the designated channel. + + Parameters + ---------- + df : pd.DataFrame + The dataframe with the ideas. + type : str + The type of ideas, either "stocks" or "crypto". + + Returns + ------- + None + """ + + # Get the database + if not util.vars.ideas_ids.empty: + # Set the types + util.vars.ideas_ids = util.vars.ideas_ids.astype( + { + "id": str, + "timestamp": "datetime64[ns]", + } + ) + + # Only keep ids that are less than 72 hours old + util.vars.ideas_ids = util.vars.ideas_ids[ + util.vars.ideas_ids["timestamp"] + > datetime.datetime.now() - datetime.timedelta(hours=72) + ] + + counter = 1 + for _, row in df.iterrows(): + if not util.vars.ideas_ids.empty: + if row["Url"] in util.vars.ideas_ids["id"].tolist(): + counter += 1 + continue + + self.add_id_to_db(row["Url"]) + + if row["Label"] == "Long": + color = 0x3CC474 + elif row["Label"] == "Short": + color = 0xE40414 + else: + color = 0x808080 + + e = discord.Embed( + title=row["Title"], + url=row["Url"], + description=row["Description"], + color=color, + timestamp=row["Timestamp"], + ) + + e.set_image(url=row["ImageURL"]) + + e.add_field( + name="Symbol", + value=row["Symbol"] if row["Symbol"] is not None else "None", + inline=True, + ) + e.add_field(name="Timeframe", value=row["Timeframe"], inline=True) + e.add_field(name="Prediction", value=row["Label"], inline=True) + + e.set_footer( + text=f"👍 {row['Likes']} | 💬 {row['Comments']}", + icon_url=data_sources["tradingview"]["icon"], + ) + + if type == "stocks": + channel = self.stocks_channel + elif type == "crypto": + channel = self.crypto_channel + elif type == "forex": + channel = self.forex_channel + + await channel.send(content=get_tagged_users([row["Symbol"]]), embed=e) + + counter += 1 + + # Only show the top 10 ideas + if counter == 11: + break + # Write to db + update_db(util.vars.ideas_ids, "ideas_ids")
+ + + @loop(hours=24) + async def crypto_ideas(self) -> None: + """ + This function posts the crypto Trading View ideas. + + Returns + ------- + None + """ + if self.crypto_channel is None: + self.crypto_channel = await get_channel( + self.bot, + config["LOOPS"]["IDEAS"]["CHANNEL"], + config["CATEGORIES"]["CRYPTO"], + ) + + df = await scraper("crypto") + await self.send_embed(df, "crypto") + + @loop(hours=24) + async def stock_ideas(self) -> None: + """ + This function posts the stocks Trading View ideas. + + Returns + ------- + None + """ + if self.stocks_channel is None: + self.stocks_channel = await get_channel( + self.bot, + config["LOOPS"]["IDEAS"]["CHANNEL"], + config["CATEGORIES"]["STOCKS"], + ) + df = await scraper("stocks") + await self.send_embed(df, "stocks") + + @loop(hours=24) + async def forex_ideas(self) -> None: + """ + This function posts the forex Trading View ideas. + + Returns + ------- + None + """ + if self.forex_channel is None: + self.forex_channel = await get_channel( + self.bot, + config["LOOPS"]["IDEAS"]["CHANNEL"], + config["CATEGORIES"]["FOREX"], + ) + df = await scraper("currencies") + await self.send_embed(df, "forex")
+ + + +
+[docs] +def crypto_parser(soup: BeautifulSoup) -> pd.DataFrame: + content = soup.find("div", class_=re.compile(r"^listContainer-")) + + # The information will be saved in these lists + titleList = [] + descriptionList = [] + labelList = [] + timeFrameList = [] # missing + symbolList = [] + timestampList = [] + commentsList = [] + imageUrlList = [] + likesList = [] + urlList = [] + + for article in content.find_all("article"): + # Get the title and description of the article + text_block = article.find("div", class_=re.compile(r"^text-block-")) + title = text_block.find("a", class_=re.compile(r"^title-")) + url = title.get("href") + title = title.text + description = text_block.find("a", class_=re.compile(r"^paragraph-")).text + + # Limit description to 4096 characters + description = description[:4096] + + # Get the image + preview = article.find("div", class_=re.compile(r"^preview-grid-")) + img = preview.find("img", class_=re.compile(r"^image-"))["src"] + symbol = preview.find("a", class_=re.compile(r"^logo-icon-"))["href"].replace( + "/symbols/", "" + )[:-1] + label = preview.find("span", class_=re.compile(r"^idea-strategy-")) + if label: + label = label.text + else: + label = "Neutral" + + # Get the other info + section = article.find("div", class_=re.compile(r"^section-")) + author = section.find("a")["data-username"] + publish_date = section.find("time")["datetime"] + publish_date = datetime.datetime.fromisoformat( + publish_date.replace("Z", "+00:00") + ) + + likes = section.find("button", class_=re.compile(r"^boostButton-")).text + comments = section.find("span", class_=re.compile(r"^content-")) + if comments: + comments = comments.get("data-overflow-tooltip-text", 0) + else: + comments = 0 + + # Append the information to the lists + titleList.append(title) + descriptionList.append(description) + labelList.append(label) + symbolList.append(symbol) + timestampList.append(publish_date) + commentsList.append(comments) + imageUrlList.append(img) + likesList.append(likes) + urlList.append(url) + timeFrameList.append("") + + data = { + "Timestamp": timestampList, + "Title": titleList, + "Description": descriptionList, + "Symbol": symbolList, + "Timeframe": timeFrameList, + "Label": labelList, + "Url": urlList, + "ImageURL": imageUrlList, + "Likes": likesList, + "Comments": commentsList, + } + + return pd.DataFrame(data)
+ + + +
+[docs] +async def scraper(type: str) -> pd.DataFrame: + """ + Extract the front page of trading ideas on TradingView. + Written by: https://github.com/mnwato/tradingview-scraper. + + Parameters + ---------- + type : string + Specify the type of trading ideas to scrape, either "stocks" or "crypto". + + Returns + ------- + pd.DataFrame + A dataframe with the ideas of the specified symbol. + """ + + if type == "crypto": + url = "https://www.tradingview.com/markets/cryptocurrencies/ideas/" + elif type == "forex": + url = "https://www.tradingview.com/markets/currencies/ideas/" + else: + url = "https://www.tradingview.com/ideas/stocks/" + + # The information will be saved in these lists + titleList = [] + descriptionList = [] + labelList = [] + timeFrameList = [] + symbolList = [] + timestampList = [] + commentsList = [] + imageUrlList = [] + likesList = [] + urlList = [] + + # Fetch the page as text + response = await get_json_data(url, text=True) + + # The response is a HTML page + soup = BeautifulSoup(response, "html.parser") + + if type == "crypto": + return crypto_parser(soup) + + # Find all divs with the following class + content = soup.find( + "div", + class_="tv-card-container__ideas tv-card-container__ideas--with-padding js-balance-content", + ) + + if content is None: + print(f"No content found for {type} ideas") + return pd.DataFrame() + + # Save the Timestamps + for time_upd in content.find_all("span", class_="tv-card-stats__time js-time-upd"): + timestampList.append( + datetime.datetime.fromtimestamp( + float(time_upd["data-timestamp"]), tz=datetime.timezone.utc + ) + ) + + for img_row in content.find_all("div", class_="tv-widget-idea__cover-wrap"): + # Get the image url + imageUrlList.append(img_row.find("img")["data-src"]) + + # Save their social info in the list + for social_row in content.find_all( + "div", class_="tv-social-row tv-widget-idea__social-row" + ): + social_info = json.loads(social_row["data-model"]) + + commentsList.append(social_info["commentsCount"]) + likesList.append(social_info["agreesCount"]) + urlList.append(f"https://www.tradingview.com{social_info['publishedUrl']}") + + # Save the titles in the list + for title_row in content.find_all("div", class_="tv-widget-idea__title-row"): + titleList.append(title_row.a.get_text()) + + for description_row in content.find_all( + "p", + class_="tv-widget-idea__description-row tv-widget-idea__description-row--clamped js-widget-idea__popup", + ): + descriptionList.append(description_row.get_text()) + + # Get the Labels, timeFrame and Symbol + for info_row in content.find_all("div", class_="tv-widget-idea__info-row"): + if "type-long" in str(info_row): + label = "Long" + elif "type-short" in str(info_row): + label = "Short" + else: + label = "Neutral" + + labelList.append(label) + + symbol_info = info_row.find("div", class_="tv-widget-idea__symbol-info") + + if symbol_info: + if symbol_info.a: + symbolList.append(symbol_info.a.text) + elif symbol_info.span: + symbolList.append(symbol_info.span.text) + else: + symbolList.append(None) + else: + symbolList.append(None) + + timeFrameList.append( + info_row.find_all("span", class_="tv-widget-idea__timeframe")[1].text + ) + + data = { + "Timestamp": timestampList, + "Title": titleList, + "Description": descriptionList, + "Symbol": symbolList, + "Timeframe": timeFrameList, + "Label": labelList, + "Url": urlList, + "ImageURL": imageUrlList, + "Likes": likesList, + "Comments": commentsList, + } + + return pd.DataFrame(data)
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(TradingView_Ideas(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/index.html b/_modules/cogs/loops/index.html new file mode 100644 index 00000000..53619771 --- /dev/null +++ b/_modules/cogs/loops/index.html @@ -0,0 +1,736 @@ + + + + + + + + + + cogs.loops.index — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.index

+# Standard libraries
+from __future__ import annotations
+
+import datetime
+
+# > Discord dependencies
+import discord
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.afterhours import afterHours
+from util.disc_util import get_channel
+from util.formatting import human_format
+from util.tv_data import tv
+from util.tv_symbols import crypto_indices, forex_indices, stock_indices
+
+# Local dependencies
+from util.vars import config, data_sources, get_json_data
+
+
+
+[docs] +class Index(commands.Cog): + """ + This class contains the cog for posting the crypto and stocks indices. + It can be enabled / disabled in the config under ["LOOPS"]["INDEX"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + if config["LOOPS"]["INDEX"]["CRYPTO"]["ENABLED"]: + self.crypto_channel = None + self.crypto_indices = [sym.split(":")[1] for sym in crypto_indices] + self.crypto.start() + + if config["LOOPS"]["INDEX"]["STOCKS"]["ENABLED"]: + self.stocks_channel = None + self.stock_indices = [sym.split(":")[1] for sym in stock_indices] + self.stocks.start() + + if config["LOOPS"]["INDEX"]["FOREX"]["ENABLED"]: + self.forex_channel = None + self.forex_indices = [sym.split(":")[1] for sym in forex_indices] + self.forex.start() + +
+[docs] + async def get_feargread(self) -> tuple[int, str] | None: + """ + Gets the last 2 Fear and Greed indices from the API. + + Returns + ------- + int + Today's Fear and Greed index. + str + The percentual change compared to yesterday's Fear and Greed index. + """ + + response = await get_json_data("https://api.alternative.me/fng/?limit=2") + + if "data" in response.keys(): + today = int(response["data"][0]["value"]) + yesterday = int(response["data"][1]["value"]) + + change = round((today - yesterday) / yesterday * 100, 2) + change = f"+{change}% 📈" if change > 0 else f"{change}% 📉" + + return today, change
+ + + @loop(hours=1) + async def crypto(self) -> None: + """ + This function will get the current prices of crypto indices on TradingView and the Fear and Greed index. + It will then post the prices in the configured channel. + + Returns + ------- + None + """ + if self.crypto_channel is None: + self.crypto_channel = await get_channel( + self.bot, + config["LOOPS"]["INDEX"]["CHANNEL"], + config["CATEGORIES"]["CRYPTO"], + ) + e = discord.Embed( + title="Crypto Indices", + description="", + color=data_sources["tradingview"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + ticker = [] + prices = [] + changes = [] + + for index in self.crypto_indices: + price, change, _, exchange, _ = await tv.get_tv_data(index, "crypto") + if price == 0: + print(index) + continue + change = round(change, 2) + change = f"+{change}% 📈" if change > 0 else f"{change}% 📉" + + if index == "TOTAL" or index == "TOTAL2" or index == "TOTAL3": + price = f"{human_format(price)}" + else: + price = f"{round(price, 2)}%" + + ticker.append( + f"[{index}](https://www.tradingview.com/symbols/{exchange}-{index}/)" + ) + prices.append(price) + changes.append(change) + + succes = await self.get_feargread() + + if succes is not None: + value, change = succes + + ticker.append( + "[Fear&Greed](https://alternative.me/crypto/fear-and-greed-index/)" + ) + prices.append(str(value)) + changes.append(change) + + ticker = "\n".join(ticker) + prices = "\n".join(prices) + changes = "\n".join(changes) + + e.add_field( + name="Index", + value=ticker, + inline=True, + ) + + e.add_field( + name="Value", + value=prices, + inline=True, + ) + + e.add_field( + name="% Change", + value=changes, + inline=True, + ) + + e.set_footer( + text="\u200b", + icon_url=data_sources["tradingview"]["icon"], + ) + + await self.crypto_channel.purge(limit=1) + await self.crypto_channel.send(embed=e) + + @loop(hours=1) + async def stocks(self) -> None: + """ + Posts the stock indices in the configured channel, only posts if the market is open. + + Returns + ------- + None + """ + if self.stocks_channel is None: + self.stocks_channel = await get_channel( + self.bot, + config["LOOPS"]["INDEX"]["CHANNEL"], + config["CATEGORIES"]["STOCKS"], + ) + # Dont send if the market is closed + if afterHours(): + return + + e = discord.Embed( + title="Stock Indices", + description="", + color=data_sources["tradingview"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + ticker = [] + prices = [] + changes = [] + + for index in self.stock_indices: + price, change, _, exchange, _ = await tv.get_tv_data(index, "stock") + if price == 0: + continue + change = round(change, 2) + change = f"+{change}% 📈" if change > 0 else f"{change}% 📉" + + if index in ["SPY", "NDX"]: + price = f"${round(price, 2)}" + # elif index == "USD10Y": + # price = f"{round(price, 2)}%" + else: + price = f"{round(price, 2)}" + + ticker.append( + f"[{index}](https://www.tradingview.com/symbols/{exchange}-{index}/)" + ) + prices.append(price) + changes.append(change) + + ticker = "\n".join(ticker) + prices = "\n".join(prices) + changes = "\n".join(changes) + + e.add_field( + name="Index", + value=ticker, + inline=True, + ) + + e.add_field( + name="Value", + value=prices, + inline=True, + ) + e.add_field( + name="% Change", + value=changes, + inline=True, + ) + + e.set_footer( + text="\u200b", + icon_url=data_sources["tradingview"]["icon"], + ) + + await self.stocks_channel.purge(limit=1) + await self.stocks_channel.send(embed=e) + + @loop(hours=1) + async def forex(self) -> None: + """ + Posts the forex indices in the configured channel, only posts if the market is open. + + Returns + ------- + None + """ + if self.forex_channel is None: + self.forex_channel = await get_channel( + self.bot, + config["LOOPS"]["INDEX"]["CHANNEL"], + config["CATEGORIES"]["FOREX"], + ) + # Dont send if the market is closed + if afterHours(): + return + + e = discord.Embed( + title="Forex Indices", + description="", + color=data_sources["tradingview"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + ticker = [] + prices = [] + changes = [] + + for index in self.forex_indices: + price, change, _, exchange, _ = await tv.get_tv_data(index, "forex") + if price == 0: + continue + change = round(change, 2) + change = f"+{change}% 📈" if change > 0 else f"{change}% 📉" + + price = f"{round(price, 2)}" + + ticker.append( + f"[{index}](https://www.tradingview.com/symbols/{exchange}-{index}/)" + ) + prices.append(price) + changes.append(change) + + if ticker == [] or prices == [] or changes == []: + return + + ticker = "\n".join(ticker) + prices = "\n".join(prices) + changes = "\n".join(changes) + + e.add_field( + name="Index", + value=ticker, + inline=True, + ) + + e.add_field( + name="Value", + value=prices, + inline=True, + ) + e.add_field( + name="% Change", + value=changes, + inline=True, + ) + + e.set_footer( + text="\u200b", + icon_url=data_sources["tradingview"]["icon"], + ) + + await self.forex_channel.purge(limit=1) + await self.forex_channel.send(embed=e)
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Index(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/liquidations.html b/_modules/cogs/loops/liquidations.html new file mode 100644 index 00000000..28be72aa --- /dev/null +++ b/_modules/cogs/loops/liquidations.html @@ -0,0 +1,852 @@ + + + + + + + + + + cogs.loops.liquidations — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.liquidations

+import glob
+import os
+import zipfile
+from concurrent.futures import ThreadPoolExecutor, as_completed
+from datetime import datetime, timedelta, timezone
+from io import BytesIO
+from xml.etree import ElementTree
+
+import discord
+import matplotlib.dates as mdates
+import matplotlib.pyplot as plt
+import pandas as pd
+import requests
+from discord.ext import commands
+from discord.ext.tasks import loop
+from matplotlib import ticker
+from tqdm import tqdm
+
+from util.disc_util import get_channel
+from util.formatting import human_format
+from util.vars import config, data_sources
+
+BACKGROUND_COLOR = "#0d1117"
+FIGURE_SIZE = (15, 7)
+COLORS_LABELS = {"#d9024b": "Shorts", "#45bf87": "Longs", "#f0b90b": "Price"}
+
+
+
+[docs] +class Liquidations(commands.Cog): + """ + This class contains the cog for posting the Liquidations chart. + It can be enabled / disabled in the config under ["LOOPS"]["LIQUIDATIONS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + if config["LOOPS"]["LIQUIDATIONS"]["ENABLED"]: + self.channel = None + self.post_liquidations.start() + + @loop(hours=24) + async def post_liquidations(self): + """ + Copy chart like https://www.coinglass.com/LiquidationData + """ + if self.channel is None: + self.channel = await get_channel( + self.bot, config["LOOPS"]["LIQUIDATIONS"]["CHANNEL"] + ) + coin = "BTCUSDT" + market = "um" + new_data = get_new_data(coin, market=market) + if new_data: + print(f"Downloaded {len(new_data)} new files.") + # Recreate the summaryf + summarize_liquidations(coin=coin, market=market) + # Load the summary + df = pd.read_csv( + f"data/summary/{coin}/{market}/liquidation_summary.csv", + index_col=0, + parse_dates=True, + ) + + if df is None or df.empty: + return + + df_price = df[["price"]].copy() + df_without_price = df.drop("price", axis=1) + df_without_price["Shorts"] = df_without_price["Shorts"] * -1 + + # This plot has 2 axes + fig, ax1 = plt.subplots() + fig.patch.set_facecolor(BACKGROUND_COLOR) + ax1.set_facecolor(BACKGROUND_COLOR) + + ax2 = ax1.twinx() + + plt.gca().xaxis.set_major_formatter(mdates.DateFormatter("%d %b")) + plt.gca().xaxis.set_major_locator(mdates.DayLocator(interval=14)) + + ax1.bar( + df_without_price.index, + df_without_price["Shorts"], + label="Shorts", + color="#d9024b", + ) + + ax1.bar( + df_without_price.index, + df_without_price["Longs"], + label="Longs", + color="#45bf87", + ) + + ax1.get_yaxis().set_major_formatter( + ticker.FuncFormatter(lambda x, _: f"${human_format(x, absolute=True)}") + ) + + # Set price axis + ax2.plot(df_price.index, df_price, color="#edba35", label="BTC Price") + ax2.set_xlim([df_price.index[0], df_price.index[-1]]) + ax2.set_ylim( + bottom=df_price.min().values * 0.95, top=df_price.max().values * 1.05 + ) + ax2.get_yaxis().set_major_formatter(lambda x, _: f"${human_format(x)}") + + # Add combined legend using the custom add_legend function + add_legend(ax2) + + # Add gridlines + plt.grid(axis="y", color="grey", linestyle="-.", linewidth=0.5, alpha=0.5) + + # Remove spines + ax1.spines["top"].set_visible(False) + ax1.spines["bottom"].set_visible(False) + ax1.spines["right"].set_visible(False) + ax1.spines["left"].set_visible(False) + ax1.tick_params(left=False, bottom=False, right=False, colors="white") + + ax2.spines["top"].set_visible(False) + ax2.spines["bottom"].set_visible(False) + ax2.spines["right"].set_visible(False) + ax2.spines["left"].set_visible(False) + ax2.tick_params(left=False, bottom=False, right=False, colors="white") + + # Fixes first and last bar not showing + ax1.set_xlim( + left=df_without_price.index[0] - timedelta(days=1), + right=df_without_price.index[-1] + timedelta(days=1), + ) + ax2.set_xlim( + left=df_without_price.index[0] - timedelta(days=1), + right=df_without_price.index[-1] + timedelta(days=1), + ) + + # Set correct size + fig.set_size_inches(FIGURE_SIZE) + + # Add the title in the top left corner + plt.text( + -0.025, + 1.125, + "Total Liquidations Chart", + transform=ax1.transAxes, + fontsize=14, + verticalalignment="top", + horizontalalignment="left", + color="white", + weight="bold", + ) + + # Convert to plot to a temporary image + file_name = "liquidations.png" + file_path = os.path.join("temp", file_name) + plt.savefig(file_path, bbox_inches="tight", dpi=300) + plt.cla() + plt.close() + + e = discord.Embed( + title="Total Liquidations", + description="", + color=data_sources["coinglass"]["color"], + timestamp=datetime.now(timezone.utc), + url="https://www.coinglass.com/LiquidationData", + ) + file = discord.File(file_path, filename=file_name) + e.set_image(url=f"attachment://{file_name}") + e.set_footer( + text="\u200b", + icon_url=data_sources["coinglass"]["icon"], + ) + + await self.channel.purge(limit=1) + await self.channel.send(file=file, embed=e) + + # Delete yield.png + os.remove(file_path)
+ + + +
+[docs] +def add_legend(ax): + # Create custom legend handles with square markers, including BTC price + legend_handles = [ + plt.Line2D( + [0], + [0], + marker="s", + color=BACKGROUND_COLOR, + markerfacecolor=color, + markersize=10, + label=label, + ) + for color, label in zip( + list(COLORS_LABELS.keys()), list(COLORS_LABELS.values()) + ) + ] + + # Add legend + legend = ax.legend( + handles=legend_handles, + loc="upper center", + bbox_to_anchor=(0.5, 1.0), + ncol=len(legend_handles), + frameon=False, + fontsize="small", + labelcolor="white", + ) + + # Make legend text bold + for text in legend.get_texts(): + text.set_fontweight("bold") + + # Adjust layout to reduce empty space around the plot + plt.subplots_adjust(left=0.05, right=0.95, top=0.875, bottom=0.1)
+ + + +
+[docs] +def get_existing_files() -> list[str]: + response = requests.get( + "https://s3-ap-northeast-1.amazonaws.com/data.binance.vision?delimiter=/&prefix=data/futures/um/daily/liquidationSnapshot/BTCUSDT/" + ) + tree = ElementTree.fromstring(response.content) + + files = [] + for content in tree.findall("{http://s3.amazonaws.com/doc/2006-03-01/}Contents"): + key = content.find("{http://s3.amazonaws.com/doc/2006-03-01/}Key").text + if key.endswith(".zip"): + files.append(key) + + return files
+ + + +
+[docs] +def extract_date_from_filename(filename: str) -> str: + return filename.split("liquidationSnapshot-")[-1].split(".")[0]
+ + + +
+[docs] +def get_local_dates(base_path: str, symbol: str, market: str): + path_pattern = os.path.join(base_path, symbol, market, "*.csv") + local_files = glob.glob(path_pattern) + local_dates = { + extract_date_from_filename(os.path.basename(file)) for file in local_files + } + return local_dates
+ + + +
+[docs] +def download_and_extract_zip( + symbol: str, date: datetime, market: str = "cm", base_extract_to="./data" +): + """ + Downloads a ZIP file from the given URL and extracts its contents to a subdirectory named after the symbol. + + Args: + symbol (str): The symbol to download data for. + date (datetime): The date for the data. + market (str): The market type. Defaults to "cm". + base_extract_to (str): The base directory to extract the contents to. Defaults to "./data". + + Returns: + None + """ + # Ensure the base_extract_to directory exists + os.makedirs(base_extract_to, exist_ok=True) + + # Create a subdirectory for the symbol + extract_to = os.path.join(base_extract_to, symbol) + os.makedirs(extract_to, exist_ok=True) + + # Subdirectory for the market + extract_to = os.path.join(extract_to, market) + os.makedirs(extract_to, exist_ok=True) + + date_str = date.strftime("%Y-%m-%d") + url = f"https://data.binance.vision/data/futures/{market}/daily/liquidationSnapshot/{symbol}/{symbol}-liquidationSnapshot-{date_str}.zip" + + try: + # Step 1: Download the ZIP file + response = requests.get(url) + response.raise_for_status() # Ensure the request was successful + + # Step 2: Extract the contents of the ZIP file + with zipfile.ZipFile(BytesIO(response.content)) as zip_ref: + zip_ref.extractall(extract_to) + + # print(f"Extracted all contents to {extract_to} for date {date_str}") + except requests.RequestException as e: + print(f"Failed to download {url}: {e}") + except zipfile.BadZipFile as e: + print(f"Failed to extract {url}: {e}")
+ + + +
+[docs] +def get_new_data( + symbol: str, market: str = "cm", base_extract_to: str = "./data" +) -> set[str]: + existing_files = get_existing_files() + existing_dates = {extract_date_from_filename(file) for file in existing_files} + + local_dates = get_local_dates(base_extract_to, symbol, market) + missing_dates = existing_dates - local_dates + + with ThreadPoolExecutor(max_workers=10) as executor: + futures = [ + executor.submit( + download_and_extract_zip, + symbol, + datetime.strptime(date, "%Y-%m-%d"), + market, + base_extract_to, + ) + for date in missing_dates + ] + if futures: + for future in tqdm( + as_completed(futures), total=len(futures), desc="Downloading files" + ): + try: + future.result() + except Exception as e: + print(f"Error occurred: {e}") + + return missing_dates
+ + + +
+[docs] +def convert_timestamp_to_date(timestamp): + return datetime.utcfromtimestamp(timestamp / 1000).strftime("%Y-%m-%d")
+ + + +
+[docs] +def summarize_liquidations(coin="BTCUSDT", market="um"): + file_pattern = f"data/{coin}/{market}/*.csv" + # Read all CSV files matching the pattern + all_files = glob.glob(file_pattern) + + df_list = [] + for file in all_files: + df = pd.read_csv(file) + df_list.append(df) + + # Concatenate all DataFrames into a single DataFrame + all_data = pd.concat(df_list, ignore_index=True) + + # Remove duplicate rows + all_data.drop_duplicates(inplace=True) + + # Convert the 'time' column to date + all_data["date"] = all_data["time"].apply(convert_timestamp_to_date) + + # Calculate total volume in USD + all_data["volume"] = all_data["original_quantity"] * all_data["average_price"] + + # Summarize the data + summary = ( + all_data.groupby(["date", "side"]) + .agg( + total_volume=("volume", "sum"), + total_liquidations=("original_quantity", "sum"), # used for avg price + ) + .reset_index() + ) + + summary["average_price"] = summary["total_volume"] / summary["total_liquidations"] + + # Pivot the summary to have separate columns for buy and sell sides + pivot_summary = summary.pivot( + index="date", columns="side", values=["total_volume", "average_price"] + ).fillna(0) + pivot_summary.columns = [ + "_".join(col).strip() for col in pivot_summary.columns.values + ] + pivot_summary = pivot_summary.rename( + columns={ + "total_volume_BUY": "Buy Volume (USD)", + "total_volume_SELL": "Sell Volume (USD)", + "average_price_BUY": "Average Buy Price", + "average_price_SELL": "Average Sell Price", + } + ) + + # Calculate overall average price + pivot_summary["Average Price"] = ( + pivot_summary["Average Buy Price"] * pivot_summary["Buy Volume (USD)"] + + pivot_summary["Average Sell Price"] * pivot_summary["Sell Volume (USD)"] + ) / (pivot_summary["Buy Volume (USD)"] + pivot_summary["Sell Volume (USD)"]) + + # Drop individual average price columns if only overall average price is needed + pivot_summary.drop( + columns=["Average Buy Price", "Average Sell Price"], inplace=True + ) + + # Rename columns as required + pivot_summary.rename( + columns={ + "Buy Volume (USD)": "Shorts", + "Sell Volume (USD)": "Longs", + "Average Price": "price", + }, + inplace=True, + ) + + # Convert the index to datetime and set it as index + pivot_summary["date"] = pd.to_datetime(pivot_summary.index) + pivot_summary = pivot_summary.set_index("date") + + # Save it locally + os.makedirs("data/summary", exist_ok=True) + os.makedirs(f"data/summary/{coin}", exist_ok=True) + os.makedirs(f"data/summary/{coin}/{market}", exist_ok=True) + pivot_summary.to_csv(f"data/summary/{coin}/{market}/liquidation_summary.csv")
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Liquidations(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/losers.html b/_modules/cogs/loops/losers.html new file mode 100644 index 00000000..acaa1965 --- /dev/null +++ b/_modules/cogs/loops/losers.html @@ -0,0 +1,481 @@ + + + + + + + + + + cogs.loops.losers — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.losers

+# Standard libraries
+import traceback
+
+# > 3rd party dependencies
+import yahoo_fin.stock_info as si
+
+# > Discord dependencies
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.afterhours import afterHours
+from util.disc_util import get_channel
+from util.formatting import format_embed
+
+# Local dependencies
+from util.vars import config
+
+
+
+[docs] +class Losers(commands.Cog): + """ + This class contains the cog for posting the top crypto and stocks losers. + It can be enabled / disabled in the config under ["LOOPS"]["LOSERS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + if config["LOOPS"]["LOSERS"]["STOCKS"]["ENABLED"]: + self.channel = None + self.losers.start() + + @loop(hours=2) + async def losers(self) -> None: + """ + If the market is open, this function posts the top 50 losers for todays stocks. + + Returns + ------- + None + """ + if self.channel is None: + self.channel = await get_channel( + self.bot, + config["LOOPS"]["LOSERS"]["CHANNEL"], + config["CATEGORIES"]["STOCKS"], + ) + + # Dont send if the market is closed + if afterHours(): + return + + try: + e = await format_embed(si.get_day_losers().head(10), "Losers", "yahoo") + await self.channel.send(embed=e) + except Exception as e: + print("Error getting or posting stock losers, error:", e) + print(traceback.format_exc())
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Losers(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/new_listings.html b/_modules/cogs/loops/new_listings.html new file mode 100644 index 00000000..4c9229cc --- /dev/null +++ b/_modules/cogs/loops/new_listings.html @@ -0,0 +1,602 @@ + + + + + + + + + + cogs.loops.new_listings — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+
+ + + +
+
+ +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.new_listings

+import asyncio
+import datetime
+
+# > Discord dependencies
+import discord
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+# Local dependencies
+from util.vars import config, get_json_data, data_sources
+from util.disc_util import get_channel
+
+
+
+[docs] +class Exchange_Listings: + """ + This class contains the cog for posting the new Binance listings + It can be enabled / disabled in the config under ["LOOPS"]["NEW_LISTINGS"]. + """ + + def __init__(self, bot: commands.Bot, exchange: str) -> None: + self.bot = bot + self.exchange = exchange + self.old_symbols = [] + self.channel = get_channel(self.bot, config["LOOPS"]["NEW_LISTINGS"]["CHANNEL"]) + + asyncio.create_task(self.set_old_symbols()) + self.new_listings.start() + +
+[docs] + async def get_symbols(self) -> list: + """ + Gets the symbols currently listed on the exchange. + + Returns + ------- + list + The symbols currently listed on the exchange + """ + + if self.exchange == "binance": + url = "https://api.binance.com/api/v3/exchangeInfo" + key1 = "symbols" + key2 = "symbol" + elif self.exchange == "kucoin": + url = "https://api.kucoin.com/api/v1/symbols" + key1 = "data" + key2 = "symbol" + elif self.exchange == "coinbase": + url = "https://api.exchange.coinbase.com/currencies" + key2 = "id" + + # Check if there have been new listings + response = await get_json_data(url) + + # Get the symbols + if self.exchange == "coinbase": + return [x[key2] for x in response] + + return [x[key2] for x in response[key1]]
+ + +
+[docs] + def create_embed(self, ticker: str) -> discord.Embed: + """ + Creates a styled embed for the newly listed ticker. + + Parameters + ---------- + ticker : str + The ticker that was newly listed. + + Returns + ------- + discord.embeds.Embed + The styled embed for the newly listed ticker. + """ + + if self.exchange == "binance": + color = data_sources["binance"]["color"] + icon_url = data_sources["binance"]["icon"] + url = f"https://www.{self.exchange}.com/en/trade/{ticker}" + elif self.exchange == "kucoin": + color = data_sources["kucoin"]["color"] + icon_url = data_sources["kucoin"]["icon"] + url = f"https://www.{self.exchange}.com/trade/{ticker}" + else: # Coinbase + color = data_sources["coinbase"]["color"] + icon_url = data_sources["coinbase"]["icon"] + url = f"https://www.pro.{self.exchange}.com/trade/{ticker}" + + e = discord.Embed( + title=f"{self.exchange.capitalize()} Lists {ticker}", + url=url, + description="", + color=color, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + # Set datetime and binance icon + e.set_footer(text="\u200b", icon_url=icon_url) + + return e
+ + +
+[docs] + async def set_old_symbols(self) -> None: + """ + Function to set the old symbols from the JSON response. + This will be used to compare the new symbols to the old symbols. + + Returns + ------- + None + """ + + # Set the old symbols + self.old_symbols = await self.get_symbols()
+ + + @loop(hours=6) + async def new_listings(self) -> None: + """ + This function will be called every 6 hours to check for new listings. + It will compare the currently listed symbols with the old symbols. + If there is a difference, it will post a message to the channel. + + Returns + ------- + None + """ + + # Get the symbols + new_symbols = await self.get_symbols() + + new_listings = [] + + if self.old_symbols == []: + await self.set_old_symbols() + + # If there is a new symbol, send a message + if len(new_symbols) > len(self.old_symbols): + new_listings = list(set(new_symbols) - set(self.old_symbols)) + + # If symbols got removed do nothing + if len(new_symbols) < len(self.old_symbols): + # Update old_symbols + self.old_symbols = new_symbols + return + + # Update old_symbols + self.old_symbols = new_symbols + + for ticker in new_listings: + await self.channel.send(embed=self.create_embed(ticker))
+ + + +
+[docs] +class Binance(commands.Cog): + """ + This class contains the cog for posting the new Binance listings + It can be enabled / disabled in the config under ["LOOPS"]["NEW_LISTINGS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + Exchange_Listings(bot, "binance")
+ + + +
+[docs] +class KuCoin(commands.Cog): + """ + This class contains the cog for posting the new KuCoin listings + It can be enabled / disabled in the config under ["LOOPS"]["NEW_LISTINGS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + Exchange_Listings(bot, "kucoin")
+ + + +
+[docs] +class CoinBase(commands.Cog): + """ + This class contains the cog for posting the new CoinBase listings + It can be enabled / disabled in the config under ["LOOPS"]["NEW_LISTINGS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + Exchange_Listings(bot, "coinbase")
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Binance(bot)) + bot.add_cog(KuCoin(bot)) + bot.add_cog(CoinBase(bot)) +
+ +
+ + + +
+ +
+ +
+
+
+ +
+ +
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/nfts.html b/_modules/cogs/loops/nfts.html new file mode 100644 index 00000000..95534277 --- /dev/null +++ b/_modules/cogs/loops/nfts.html @@ -0,0 +1,936 @@ + + + + + + + + + + cogs.loops.nfts — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.nfts

+## > Imports
+# > Standard library
+import datetime
+import re
+
+# > Discord dependencies
+import discord
+import numpy as np
+
+# > Third party
+import pandas as pd
+from bs4 import BeautifulSoup
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.cg_data import cg
+from util.disc_util import get_channel
+from util.formatting import format_change
+
+# > Local
+from util.vars import config, data_sources, get_json_data
+
+
+
+[docs] +class NFTS(commands.Cog): + """ + This class contains the cog for posting the top NFTs. + It can be configured in the config.yaml file under ["LOOPS"]["NFTS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + if config["LOOPS"]["NFTS"]["ENABLED"]: + if config["LOOPS"]["NFTS"]["TOP"]: + self.top_channel = None + self.top_nfts.start() + + if config["LOOPS"]["NFTS"]["UPCOMING"]: + self.upcoming_channel = None + self.upcoming_nfts.start() + + if config["LOOPS"]["NFTS"]["P2E"]: + self.p2e_channel = None + self.top_p2e.start() + + if config["LOOPS"]["TRENDING"]["NFTS"]: + self.trending_channel = None + self.trending_nfts.start() + + @loop(hours=1) + async def top_nfts(self): + if self.top_channel is None: + self.top_channel = await get_channel( + self.bot, + config["LOOPS"]["NFTS"]["TOP"]["CHANNEL"], + config["CATEGORIES"]["NFTS"], + ) + opensea_top = await get_opensea() + cmc_top = await top_cmc() + + await self.top_channel.purge(limit=2) + + for df, name in [(opensea_top, "Opensea"), (cmc_top, "CoinMarketCap")]: + if df.empty: + print("No top NFTs found for " + name) + return + + if "symbol" not in df.columns: + return + + if name == "Opensea": + url = "https://opensea.io/rankings" + color = data_sources["opensea"]["color"] + icon_url = data_sources["opensea"]["icon"] + elif name == "CoinMarketCap": + url = "https://coinmarketcap.com/nft/collections/" + color = data_sources["coinmarketcap"]["color"] + icon_url = data_sources["coinmarketcap"]["icon"] + + e = discord.Embed( + title=f"Top {len(df)} {name} NFTs", + url=url, + description="", + color=color, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.add_field( + name="NFT", + value="\n".join(df["symbol"].tolist()), + inline=True, + ) + + e.add_field( + name="Price", + value="\n".join(df["price"].tolist()), + inline=True, + ) + + e.add_field( + name="Volume", + value="\n".join(df["volume"].astype(str).tolist()), + inline=True, + ) + + # Set empty text as footer, so we can see the icon + e.set_footer(text="\u200b", icon_url=icon_url) + + await self.top_channel.send(embed=e) + + @loop(hours=1) + async def trending_nfts(self): + if self.trending_channel is None: + self.trending_channel = await get_channel( + self.bot, + config["LOOPS"]["TRENDING"]["CHANNEL"], + config["CATEGORIES"]["NFTS"], + ) + await self.trending_channel.purge(limit=2) + + await self.opensea_trending() + await self.gc_trending() + + + + + + + + @loop(hours=1) + async def upcoming_nfts(self): + if self.upcoming_channel is None: + self.upcoming_channel = await get_channel( + self.bot, + config["LOOPS"]["NFTS"]["UPCOMING"]["CHANNEL"], + config["CATEGORIES"]["NFTS"], + ) + upcoming = await upcoming_cmc() + + if upcoming.empty: + print("No upcoming NFTs found") + return + + if "symbol" not in upcoming.columns: + return + + upcoming = upcoming.head(10) + + e = discord.Embed( + title=f"Top {len(upcoming)} Upcoming NFTs", + url="https://coinmarketcap.com/nft/upcoming/", + description="", + color=data_sources["coinmarketcap"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.add_field( + name="NFT", + value="\n".join(upcoming["symbol"].tolist()), + inline=True, + ) + + e.add_field( + name="Price", + value="\n".join(upcoming["price"].tolist()), + inline=True, + ) + + e.add_field( + name="Drop Type", + value="\n".join(upcoming["dropType"].tolist()), + inline=True, + ) + e.set_footer(text="\u200b", icon_url=data_sources["coinmarketcap"]["icon"]) + + await self.upcoming_channel.purge(limit=1) + await self.upcoming_channel.send(embed=e) + + @loop(hours=1) + async def top_p2e(self): + if self.p2e_channel is None: + self.p2e_channel = await get_channel( + self.bot, + config["LOOPS"]["NFTS"]["P2E"]["CHANNEL"], + config["CATEGORIES"]["NFTS"], + ) + try: + p2e = await p2e_games() + except Exception as e: + print("Error fetching PlayToEarn data: ", e) + return + + if p2e.empty: + return + + url = "https://playtoearn.net/blockchaingames/All-Blockchain/All-Genre/All-Status/All-Device/NFT/nft-crypto-PlayToEarn/nft-required-FreeToPlay" + + e = discord.Embed( + title=f"Top {len(p2e)} Blockchain Games", + url=url, + description="", + color=data_sources["playtoearn"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.add_field( + name="Game", + value="\n".join(p2e["name"].tolist()), + inline=True, + ) + + e.add_field( + name="Social 24h", + value="\n".join(p2e["social"].tolist()), + inline=True, + ) + + e.add_field( + name="Status", + value="\n".join(p2e["status"].tolist()), + inline=True, + ) + + e.set_footer( + text="\u200b", + icon_url=data_sources["playtoearn"]["icon"], + ) + + await self.p2e_channel.purge(limit=1) + await self.p2e_channel.send(embed=e)
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(NFTS(bot)) + + +
+[docs] +async def get_opensea(url=""): + """ + _summary_ + + Parameters + ---------- + url : str, optional + Can be either "trending" or empty, by default "" + + Returns + ------- + _type_ + _description_ + """ + + html_doc = await get_json_data( + f"https://opensea.io/rankings/{url}", + headers={"User-Agent": "Mozilla/5.0"}, + text=True, + ) + + html_doc = html_doc[html_doc.find(':pageInfo"}},') + len(':pageInfo"}},') :] + html_doc = html_doc[: html_doc.find(":edges:10")] + + rows = html_doc.split('"node":{') + + opensea_nfts = [] + + for row in rows[1:]: + nft_dict = {} + + name = re.search(r"\"name\":\"(.*?)\"", row).group(1) + slug = re.search(r"\"slug\":\"(.*?)\"", row) + + if slug: + slug = slug.group(1) + else: + slug = "" + + price_data = re.findall(r"\"unit\":\"(.*?)\"", row) + change = re.search(r"\"volumeChange\":(.*?),", row) + symbol = re.search(r"\"symbol\":\"(.*?)\"", row).group(1) + + if len(price_data) == 2: + floor_price = f"{round(float(price_data[0]),3)} {symbol}" + volume = price_data[1] + else: + floor_price = "?" + volume = price_data[0] + + volume = f"{int(float(volume))} {symbol}" + change = float(change.group(1)) * 100 + + if change != 0: + if change > 1: + change = int(change) + else: + change = round(change, 2) + volume = f"{volume} ({format_change(change)})" + + nft_dict["symbol"] = f"[{name}](https://opensea.io/collection/{slug})" + nft_dict["price"] = floor_price + nft_dict["volume"] = volume + + opensea_nfts.append(nft_dict) + + return pd.DataFrame(opensea_nfts)
+ + + +
+[docs] +async def top_cmc(): + data = await get_json_data( + "https://api.coinmarketcap.com/nft/v3/nft/collectionsv2?start=0&limit=100&category=&collection=&blockchain=&sort=volume&desc=true&period=1" + ) + + # Convert to dataframe + if "data" not in data: + print("No data found in CoinMarketCap response") + return pd.DataFrame() + if "collections" not in data["data"]: + print("No collections found in CoinMarketCap response") + return pd.DataFrame() + + df = pd.DataFrame(data["data"]["collections"]) + + df = df.head(10) + + # Unpack all oneDay data + df = pd.concat([df.drop(["oneDay"], axis=1), df["oneDay"].apply(pd.Series)], axis=1) + + # name, url, price, volume, volume change + # Conditionally concatenate "name" and "website" only when "website" is not NaN + df["symbol"] = np.where( + df["website"].notna() & (df["website"] != ""), + "[" + df["name"] + "]" + "(" + df["website"] + ")", + df["name"], + ) + df["price"] = df["floorPriceUsd"].apply(lambda x: f"${x:,.2f}") + df["change"] = df["averagePriceChangePercentage"].apply(lambda x: format_change(x)) + df["price"] = df["price"] + " (" + df["change"] + ")" + df["volume"] = df["volume"].apply(lambda x: f"{x:,.0f} ETH") + df["volume_change"] = df["volumeChangePercentage"].apply(lambda x: format_change(x)) + df["volume"] = df["volume"] + " (" + df["volume_change"] + ")" + + return df
+ + + +
+[docs] +async def upcoming_cmc(): + # Could remove category and expire from URL + data = await get_json_data( + "https://api.coinmarketcap.com/nft/v3/nft/upcoming-drops?start=0&limit=20&category=Popular&expire=30" + ) + + # Convert the data to a pandas DataFrame + df = pd.DataFrame(data["data"]["data"]) + + df = df.head(10) + + # name, websiteUrl, price, dropDate + # Filter out the columns that actually exist in the DataFrame + existing_columns = [ + col for col in ["name", "websiteUrl", "price", "dropType"] if col in df.columns + ] + + # Use only the existing columns to filter the DataFrame + df = df[existing_columns] + + # Use same method as #events channel time + # Rename to start_time + # df["start_time"] = df["dropDate"].apply( + # lambda x: f"<t:{int(x/1000)}:d>" if pd.notnull(x) else "" + # ) + + # Conditionally concatenate "name" and "website" only when "website" is not NaN + df["symbol"] = np.where( + df["websiteUrl"].notna() & (df["websiteUrl"] != ""), + "[" + df["name"] + "]" + "(" + df["websiteUrl"] + ")", + df["name"], + ) + + return df
+ + + +
+[docs] +async def p2e_games(): + URL = "https://playtoearn.net/blockchaingames/All-Blockchain/All-Genre/All-Status/All-Device/NFT/nft-crypto-PlayToEarn/nft-required-FreeToPlay" + + html = await get_json_data(URL, text=True) + soup = BeautifulSoup(html, "html.parser") + items = soup.find("table", class_="table table-bordered mainlist") + + if items is None: + return pd.DataFrame() + + allItems = items.find_all("tr") + + p2e_games = [] + + # Skip header + ad + iterator = 2 + for iterator in range(2, 12): + data = {} + + allItems_td = allItems[iterator].find_all("td") + if len(allItems_td) < 11: + continue + + name = allItems_td[2].find("div", class_="dapp_name").find_next("span").text + url = allItems_td[2].find_next("a")["href"] + status = allItems_td[6].get_text("title") + social_24h_change = allItems_td[10].find_all("span") + social_24h = social_24h_change[0].text + if len(social_24h_change) > 1: + social_change = social_24h_change[1].text.replace("%", "").replace(",", "") + else: + social_change = 0 + + data["name"] = f"[{name}]({url})" + data["status"] = status + data["social"] = f"{social_24h} ({format_change(float(social_change))})" + + p2e_games.append(data) + + return pd.DataFrame(p2e_games)
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/option_alert.html b/_modules/cogs/loops/option_alert.html new file mode 100644 index 00000000..0a0a18d6 --- /dev/null +++ b/_modules/cogs/loops/option_alert.html @@ -0,0 +1,711 @@ + + + + + + + + + + cogs.loops.option_alert — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+
+ + + +
+
+ +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.option_alert

+# Standard libraries
+import datetime
+import pandas as pd
+import time
+import inspect
+
+# > Discord dependencies
+import discord
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+# Local dependencies
+import util.vars
+from util.vars import config, get_json_data
+from util.disc_util import get_channel, get_tagged_users, get_guild
+from util.afterhours import afterHours
+from util.db import clean_old_db, merge_and_update
+from cogs.loops.options import get_UW_data
+
+
+
+[docs] +class Option_alert(commands.Cog): + """ + This class contains the cog for posting the latest Unusual Whales alerts. + It can be enabled / disabled in the config under ["LOOPS"]["UNUSUAL_WHALES"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + self.emoji_dict = {} + self.guild = get_guild(bot) + + if config["LOOPS"]["OPTION_ALERT"]["ENABLED"]: + self.alerts_channel = get_channel( + self.bot, config["LOOPS"]["OPTION_ALERT"]["CHANNEL"] + ) + + self.overview_channel = get_channel( + self.bot, + config["LOOPS"]["OPTION_ALERT"]["OVERVIEW_CHANNEL"], + config["CATEGORIES"]["OPTIONS"], + ) + + self.token = config["LOOPS"]["OPTION_ALERT"]["TOKEN"] + self.alerts.start() + + @loop(minutes=5) + async def alerts(self) -> None: + """ + This function posts the Unusual Whales alerts on Discord. + + Returns + ------- + None + """ + + # Check if the market is open + if afterHours(): + return + + # Get the emojis if not already done + if self.emoji_dict == {}: + # Get the emojis and store them in emoji_dict + self.emoji_dict = await get_json_data( + "https://phx.unusualwhales.com/api/tags/all", + { + "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.54 Safari/537.36" + }, + ) + + # start_date and expiry_start_data depends on how often the function is called + last_5_min = int((time.time() - (5 * 60)) * 1000) + + # Check the last 5 minutes on the API + url = f"https://phx.unusualwhales.com/api/option_quotes?offset=0&sort=timestamp&search=&sector=&tag=&end_date=9999999999999&start_date={last_5_min}&expiry_start_date={last_5_min}&expiry_end_date=9999999999999&min_ask=0&max_ask=9999999999999&volume_direction=desc&expiry_direction=desc&alerted_direction=desc&oi_direction=desc&normal=true" + + # Use the token in the header + headers = { + "authorization": self.token, + "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.54 Safari/537.36", + } + + df = await get_UW_data(url, headers) + + if df.empty: + return + + # Only keep the important information + df = df[ + [ + "alert_time", + "id", + "ticker_symbol", + "option_type", + "strike_price", # Also named underlying + "expires_at", + "stock_price", + "bid", + "ask", + "min_ask", + "max_ask", + "volume", + "implied_volatility", + "sector", + "tags", + "tier", + "is_recommended", + "open_interest", + "delta", + "theta", + ] + ] + + # Calculate the percentual difference between current price and strike price + df["strike_price"] = df["strike_price"].astype(float) + df["stock_price"] = df["stock_price"].astype(float) + df["difference"] = ( + (df["strike_price"] - df["stock_price"]) / df["stock_price"] * 100 + ) + df["difference"] = df["difference"].round(2) + df["difference"] = df["difference"].astype(str) + "%" + + # Convert IV to percent + df["implied_volatility"] = df["implied_volatility"].astype(float) + df["IV"] = df["implied_volatility"] * 100 + df["IV"] = df["IV"].round(2) + df["IV"] = df["IV"].astype(str) + "%" + + # Round theta and delta + df["theta"] = df["theta"].astype(float) + df["theta"] = df["theta"].round(3) + df["delta"] = df["delta"].astype(float) + df["delta"] = df["delta"].round(3) + + # For each ticker in the df send a message + for _, row in df.iterrows(): + # Only use the first letter of the option type + option_type = row["option_type"][0].upper() + + emojis = "" + for tag in row["tags"]: + try: + emojis += self.emoji_dict[tag]["emoji"] + # In case emoji_dict is empty + except KeyError: + print(f"Could not find emoji for {tag}") + + # Create the embed + e = discord.Embed( + title=f"${row['ticker_symbol']} {row['expires_at']} {option_type} ${row['strike_price']}", + url=f"https://unusualwhales.com/alerts/{row['id']}", + # Use inspect.cleandoc() to remove the indentation + description=inspect.cleandoc( + f""" + {emojis} + Bid-Ask: ${row['bid']} - ${row['ask']} + Interest: {row['open_interest']} + Volume: {row['volume']} + IV: {row['IV']} + % Diff: {row["difference"]} + Underlying: ${row['stock_price']} + Θ | Δ: {row['theta']} | {row['delta']} + Sector: {row['sector']} + Tier: {row['tier']} + Recommended: {row['is_recommended']} + {emojis} + """ + ), + color=0xE40414 if option_type == "P" else 0x3CC474, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.set_footer( + # Use the time the alert was created in the footer + text=f"Alerted at {row['alert_time']}", + icon_url="https://docs.unusualwhales.com/images/banner.png", + ) + + await self.alerts_channel.send( + content=get_tagged_users([row["ticker_symbol"]]), embed=e + ) + + # Add the data to the database + update_options_db( + row["ticker_symbol"], + row["expires_at"], + option_type, + row["strike_price"], + row["volume"], + emojis, + ) + + await self.options_overview() + +
+[docs] + async def options_overview(self): + if util.vars.options_db.empty: + return + + # Gather the data for the summary + num_calls = len( + util.vars.options_db[util.vars.options_db["option_type"] == "C"] + ) + num_puts = len(util.vars.options_db[util.vars.options_db["option_type"] == "P"]) + + num_bears = len( + util.vars.options_db[util.vars.options_db["bull/bear"] == "bear"] + ) + num_bulls = len( + util.vars.options_db[util.vars.options_db["bull/bear"] == "bull"] + ) + + # Top row of the embed shows an summary of P/C ratio, Bullish/Bearish + e = discord.Embed( + title=f"Options Overview", + description="", + color=self.guild.self_role.color, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + most_mentioned = util.vars.options_db["ticker"].value_counts() + + e.add_field(name="Calls - Puts", value=f"{num_calls} - {num_puts}", inline=True) + e.add_field( + name="Bullish - Bearish", value=f"{num_bulls} - {num_bears}", inline=True + ) + e.add_field( + name="Most Active Ticker", + value=f"{most_mentioned.index[0]} ({most_mentioned.iloc[0]})", + inline=True, + ) + + # Sort by volume + util.vars.options_db["volume"] = util.vars.options_db["volume"].astype(int) + util.vars.options_db = util.vars.options_db.sort_values( + by=["volume"], ascending=False + ) + + # First show the top 10 bullish options, ranked by count and volume + bullish = util.vars.options_db[util.vars.options_db["bull/bear"] == "bull"] + + # Then show the top 10 bearish options + bearish = util.vars.options_db[util.vars.options_db["bull/bear"] == "bear"] + + if not bullish.empty: + bull_counts, bull_options, bull_volumes = self.get_top20(bullish) + e.add_field(name="Bullish", value="\n".join(bull_counts), inline=True) + e.add_field(name="Options", value="\n".join(bull_options), inline=True) + e.add_field(name="Volume", value="\n".join(bull_volumes), inline=True) + + if not bearish.empty: + bear_counts, bear_options, bear_volumes = self.get_top20(bearish) + e.add_field(name="Bearish", value="\n".join(bear_counts), inline=True) + e.add_field(name="Options", value="\n".join(bear_options), inline=True) + e.add_field(name="Volume", value="\n".join(bear_volumes), inline=True) + + await self.overview_channel.purge(limit=1) + await self.overview_channel.send(embed=e)
+ + +
+[docs] + def get_top20(self, df): + counts = [] + options = [] + volumes = [] + + for _, row in df.head(20).iterrows(): + counts.append(f"{row['ticker']}") + options.append( + f"{row['expiration']} {row['option_type']} ${row['strike_price']}" + ) + volumes.append(str(row["volume"])) + + return counts, options, volumes
+
+ + + +
+[docs] +def update_options_db(ticker, expiration, option_type, strike, volume, emojis): + if "🐻" in emojis: + emoji = "bear" + elif "🐂" in emojis: + emoji = "bull" + else: + emoji = "none" + + option_dict = { + "ticker": ticker, + "expiration": expiration, + "option_type": option_type, + "strike_price": strike, + "volume": volume, + "bull/bear": emoji, + } + + # Convert it to a dataframe + option_db = pd.DataFrame([option_dict]) + + # Add timestamp + option_db["timestamp"] = datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S") + + # Clean the old db + util.vars.options_db = clean_old_db(util.vars.options_db, 1) + util.vars.options_db = merge_and_update(util.vars.options_db, option_db, "options")
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Option_alert(bot)) +
+ +
+ + + +
+ +
+ +
+
+
+ +
+ +
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/options.html b/_modules/cogs/loops/options.html new file mode 100644 index 00000000..b6ac4dc8 --- /dev/null +++ b/_modules/cogs/loops/options.html @@ -0,0 +1,599 @@ + + + + + + + + + + cogs.loops.options — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+
+ + + +
+
+ +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.options

+# Standard libraries
+import datetime
+import pandas as pd
+
+# > Discord dependencies
+import discord
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+# Local dependencies
+from util.vars import config, get_json_data, data_sources
+from util.disc_util import get_channel, get_guild
+from util.formatting import human_format
+
+
+
+[docs] +async def get_UW_data(url, overwrite_headers=None, last_15min=False): + if not overwrite_headers: + headers = { + "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.54 Safari/537.36", + } + else: + headers = overwrite_headers + + data = await get_json_data(url, headers) + df = pd.DataFrame(data) + + if df.empty: + return df + + # Get the timestamp convert to datetime and local time + df["alert_time"] = pd.to_datetime(df["timestamp"], utc=True) + + # Filter df on last 15 minutes + if last_15min: + df = df[ + df["alert_time"] + > datetime.datetime.now(datetime.timezone.utc) + - datetime.timedelta(minutes=15) + ] + + if df.empty: + return df + + df["alert_time"] = df["alert_time"].dt.tz_convert( + datetime.datetime.now(datetime.timezone.utc).astimezone().tzinfo + ) + + # Format to string + df["alert_time"] = df["alert_time"].dt.strftime("%I:%M %p") + + return df
+ + + +
+[docs] +class Options(commands.Cog): + """ + This class contains the cog for posting the latest Unusual Whales alerts. + It can be enabled / disabled in the config under ["LOOPS"]["UNUSUAL_WHALES"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + self.guild = get_guild(bot) + + self.volume_channel = get_channel( + self.bot, config["LOOPS"]["OPTIONS"]["VOLUME_CHANNEL"] + ) + + self.spacs_channel = get_channel( + self.bot, config["LOOPS"]["OPTIONS"]["SPACS_CHANNEL"] + ) + + self.shorts_channel = get_channel( + self.bot, config["LOOPS"]["OPTIONS"]["SHORTS_CHANNEL"] + ) + + self.volume.start() + self.spacs.start() + self.shorts.start() + +
+[docs] + def make_UW_embed(self, row): + e = discord.Embed( + title=f"${row['ticker_symbol']}", + url=f"https://unusualwhales.com/stock/{row['ticker_symbol']}", + description="", + color=self.guild.self_role.color, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.add_field( + name="Volume", value=f"${human_format(float(row['volume']))}", inline=True + ) + e.add_field( + name="Average 30d Volume", + value=f"${human_format(float(row['avg_volume_last_30_days']))}", + inline=True, + ) + e.add_field( + name="Volume Deviation", + value=f"{round(float(row['volume_dev_from_norm']))}", + inline=True, + ) + e.add_field(name="Price", value=f"${row['bid_price']}", inline=True) + + e.set_footer( + # Use the time the alert was created in the footer + text=f"Alerted at {row['alert_time']}", + icon_url=data_sources["unusualwhales"]["icon"], + ) + + return e
+ + + @loop(minutes=15) + async def volume(self): + url = "https://phx.unusualwhales.com/api/stock_feed" + df = await get_UW_data(url, last_15min=True) + + if not df.empty: + # Iterate over each row and post the alert + for _, row in df.iterrows(): + e = self.make_UW_embed(row) + await self.volume_channel.send(embed=e) + + @loop(minutes=15) + async def spacs(self): + url = "https://phx.unusualwhales.com/api/warrant_alerts" + df = await get_UW_data(url, last_15min=True) + + if not df.empty: + # Iterate over each row and post the alert + for _, row in df.iterrows(): + e = self.make_UW_embed(row) + await self.spacs_channel.send(embed=e) + + @loop(hours=24) + async def shorts(self): + url = "https://phx.unusualwhales.com/api/short_interest_low" + headers = { + "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.54 Safari/537.36", + } + + data = await get_json_data(url, headers) + df = pd.DataFrame(data["data"]) + + if df.empty: + return + + # Cast to float + df["short_interest"] = df["short_interest"].astype(float) + + # Sort on short_interest + df = df.sort_values(by="short_interest", ascending=False) + df["float_shares"] = df["float_shares"].astype(float) + df["float_shares"] = df["float_shares"].apply(lambda x: human_format(x)) + + df["outstanding"] = df["outstanding"].astype(float) + df["outstanding"] = df["outstanding"].apply(lambda x: human_format(x)) + + df["short_interest"] = df["short_interest"].astype(str) + + # Combine both in 1 string + df["float - outstanding"] = df["float_shares"] + " - " + df["outstanding"] + + top20 = df.head(20) + + symbols = "\n".join(top20["symbol"].tolist()) + float_oustanding = "\n".join(top20["float - outstanding"].tolist()) + short_interest = "\n".join(top20["short_interest"].tolist()) + + # Only show the top 20 as embed + e = discord.Embed( + title=f"Top Short Interest Reported", + url=f"https://unusualwhales.com/shorts", + description="", + color=self.guild.self_role.color, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.set_footer( + # Use the time the alert was created in the footer + text="\u200b", + icon_url=data_sources["unusualwhales"]["icon"], + ) + + e.add_field(name="Symbol", value=symbols, inline=True) + e.add_field(name="Float - Outstanding", value=float_oustanding, inline=True) + e.add_field(name="Short Interest", value=short_interest, inline=True) + + await self.shorts_channel.purge(limit=1) + await self.shorts_channel.send(embed=e)
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Options(bot)) +
+ +
+ + + +
+ +
+ +
+
+
+ +
+ +
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/overview.html b/_modules/cogs/loops/overview.html new file mode 100644 index 00000000..3a755c58 --- /dev/null +++ b/_modules/cogs/loops/overview.html @@ -0,0 +1,649 @@ + + + + + + + + + + cogs.loops.overview — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.overview

+## > Imports
+# > Standard libraries
+import datetime
+from collections import Counter, defaultdict
+
+# > Discord dependencies
+import discord
+from discord.ext.tasks import loop
+
+# Local dependencies
+import util.vars
+from util.disc_util import get_channel, get_guild
+from util.formatting import format_change
+from util.vars import config, get_json_data
+
+text_to_emoji = defaultdict(lambda: "🦆", {"bear": "🐻", "bull": "🐂", "neutral": "🦆"})
+
+
+
+[docs] +class Overview: + """ + This class contains the cog for posting the top crypto and stocks mentions. + It can be configured in the config.yaml file under ["LOOPS"]["OVERVIEW"]. + """ + + def __init__(self, bot): + self.bot = bot + self.guild = get_guild(bot) + self.global_crypto = {} + self.global_stocks = {} + + self.global_overview.start() + + if config["LOOPS"]["OVERVIEW"]["STOCKS"]["ENABLED"]: + self.stocks_channel = None + self.do_stocks = True + else: + self.do_stocks = False + + if config["LOOPS"]["OVERVIEW"]["CRYPTO"]["ENABLED"]: + self.crypto_channel = None + self.do_crypto = True + else: + self.do_crypto = False + +
+[docs] + async def overview(self, category, tickers, sentiment): + # Make sure that the new db is not empty + if not util.vars.tweets_db.empty: + if self.do_stocks and category == "stocks": + await self.make_overview(category, tickers, sentiment) + if self.do_crypto and category == "crypto": + await self.make_overview(category, tickers, sentiment)
+ + + @loop(minutes=5) + async def global_overview(self): + if util.vars.tweets_db.empty: + return + + categories = [] + if self.do_stocks: + categories.append("stocks") + if self.do_crypto: + categories.append("crypto") + + for category in categories: + db = util.vars.tweets_db.loc[util.vars.tweets_db["category"] == category] + + if db.empty: + return + + # Get the top 50 mentions + top50 = db["ticker"].value_counts()[:50] + + for ticker, _ in top50.items(): + # Get the global tweets about the ticker using the API + if category == "stocks": + global_mentions = None # await count_tweets(ticker) + if global_mentions is not None: + self.global_stocks[ticker] = global_mentions + elif category == "crypto": + global_mentions = None # await count_tweets(ticker) + if global_mentions is not None: + self.global_crypto[ticker] = await count_tweets(ticker) + +
+[docs] + async def make_overview(self, category: str, tickers: list, last_sentiment: str): + if self.stocks_channel is None: + self.stocks_channel = await get_channel( + self.bot, + config["LOOPS"]["OVERVIEW"]["CHANNEL"], + config["CATEGORIES"]["STOCKS"], + ) + if self.crypto_channel is None: + self.crypto_channel = await get_channel( + self.bot, + config["LOOPS"]["OVERVIEW"]["CHANNEL"], + config["CATEGORIES"]["CRYPTO"], + ) + + # Post the overview for stocks and crypto + db = util.vars.tweets_db.loc[util.vars.tweets_db["category"] == category] + + if db.empty: + return + + # Get the top 50 mentions + top50 = db["ticker"].value_counts()[:50] + + # Make the list for embeds + count_list = [] + ticker_list = [] + sentiment_list = [] + + # Add overview of sentiment for each ticker + for ticker, count in top50.items(): + # Get the sentiment for the ticker + sentiment = db.loc[db["ticker"] == ticker]["sentiment"].tolist() + + change = db.loc[db["ticker"] == ticker]["change"].tolist()[0] + change = change.replace("%", "").replace("+", "") + + try: + change = format_change(float(change)) + except ValueError: + change = "" # Do not specify it + + # Convert sentiment into a single str, i.e. "6🐂 2🦆 2🐻" + sentiment = [text_to_emoji[sent] for sent in sentiment] + sentiment = dict(Counter(sentiment)) + + formatted_sentiment = "" + # Use this method to sort the dict + for emoji in ["🐂", "🦆", "🐻"]: + if emoji in sentiment.keys(): + if emoji == last_sentiment and ticker in tickers: + formatted_sentiment += f"**{sentiment[emoji]}**{emoji} " + else: + formatted_sentiment += f"{sentiment[emoji]}{emoji} " + + if category == "stocks": + if ticker in self.global_stocks.keys(): + count = f"{count} - {self.global_stocks[ticker]}" + + if category == "crypto": + if ticker in self.global_crypto.keys(): + count = f"{count} - {self.global_crypto[ticker]}" + + if ticker in tickers: + # Make bold + ticker = f"**{ticker} ({change})**" + count = f"**{count}**" + else: + ticker = f"{ticker} ({change})" + + # Add count, symbol, sentiment to embed lists + count_list.append(str(count)) + ticker_list.append(ticker) + sentiment_list.append(formatted_sentiment) + + # Make the embed + e = discord.Embed( + title=f"Top {category.capitalize()} Mentions Of The Last 24 Hours", + description="", + color=self.guild.self_role.color, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.add_field( + name="Mentions", + value="\n".join(count_list), + inline=True, + ) + + e.add_field( + name="Ticker", + value="\n".join(ticker_list), + inline=True, + ) + + e.add_field( + name="Sentiment", + value="\n".join(sentiment_list), + inline=True, + ) + + if category == "crypto": + # Delete previous message + await self.crypto_channel.purge(limit=1) + await self.crypto_channel.send(embed=e) + else: + await self.stocks_channel.purge(limit=1) + await self.stocks_channel.send(embed=e)
+
+ + + +
+[docs] +async def count_tweets(ticker: str) -> int: + """ + Counts the number of tweets for a ticker during the last 24 hours. + https://developer.twitter.com/en/docs/twitter-api/tweets/counts/api-reference/get-tweets-counts-recent + Max 300 requests per 15 minutes, so 20 requests per minute. + + Parameters + ---------- + ticker : str + The ticker to count the tweets for. + + Returns + ------- + int + Returns the number of tweets for the ticker. + """ + + # Count the last 24 hours + # Can add -is:retweet in query param to exclude retweets + start_time = ( + datetime.datetime.utcnow() - datetime.timedelta(days=1) + ).isoformat() + "Z" + url = f"https://api.twitter.com/2/tweets/counts/recent?query={ticker}&granularity=day&start_time={start_time}" + counts = await get_json_data(url=url, headers={"Authorization": f"Bearer {None}"}) + + if "meta" in counts.keys(): + if "total_tweet_count" in counts["meta"].keys(): + return counts["meta"]["total_tweet_count"]
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/reddit.html b/_modules/cogs/loops/reddit.html new file mode 100644 index 00000000..5b0aa61e --- /dev/null +++ b/_modules/cogs/loops/reddit.html @@ -0,0 +1,783 @@ + + + + + + + + + + cogs.loops.reddit — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.reddit

+import html
+import os
+import re
+from datetime import datetime, timedelta
+
+import asyncpraw
+import pandas as pd
+from discord import Embed
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+import util.vars
+from util.db import update_db
+from util.disc_util import get_channel, get_webhook
+from util.vars import config, data_sources
+
+URL_REGEX = r"(?P<url>https?://[^\s]+)"
+MARKDOWN_LINK_REGEX = r"\[(?P<text>[^\]]+)\]\((?P<url>https?://[^\s]+)\)"
+
+
+
+[docs] +class Reddit(commands.Cog): + """ + This class contains the cog for posting the top reddit posts. + It can be enabled / disabled in the config under ["LOOPS"]["REDDIT"]. + """ + + def __init__(self, bot: commands.bot.Bot) -> None: + self.bot = bot + self.first_time = True + + self.reddit = asyncpraw.Reddit( + client_id=os.getenv("REDDIT_PERSONAL_USE"), + client_secret=os.getenv("REDDIT_SECRET"), + user_agent=os.getenv("REDDIT_APP_NAME"), + username=os.getenv("REDDIT_USERNAME"), + password=os.getenv("REDDIT_PASSWORD"), + ) + + if config["LOOPS"]["REDDIT"]["WALLSTREETBETS"]["ENABLED"]: + self.wsb_channel = None + self.wsb_scraper.start() + + if config["LOOPS"]["REDDIT"]["CRYPTOMOONSHOTS"]["ENABLED"]: + self.cmc_channel = None + self.cms_scraper.start() + +
+[docs] + def add_id_to_db(self, id: str) -> None: + """ + Adds the given id to the database. + """ + + util.vars.reddit_ids = pd.concat( + [ + util.vars.reddit_ids, + pd.DataFrame( + [ + { + "id": id, + "timestamp": datetime.now(), + } + ] + ), + ], + ignore_index=True, + )
+ + + @loop(hours=12) + async def wsb_scraper(self): + if self.wsb_channel is None: + self.wsb_channel = await get_channel( + self.bot, config["LOOPS"]["REDDIT"]["WALLSTREETBETS"]["CHANNEL"] + ) + await self.reddit_scraper(subreddit_name="WallStreetBets") + self.first_time = False + + # To prevent it from going to quick + if not self.first_time: + await self.reddit_scraper(subreddit_name="WallStreetBets") + + @loop(hours=12) + async def cms_scraper(self): + if self.cms_scraper is None: + self.cmc_channel = await get_channel( + self.bot, config["LOOPS"]["REDDIT"]["CRYPTOMOONSHOTS"]["CHANNEL"] + ) + await self.reddit_scraper(subreddit_name="CryptoMoonShots") + self.first_time = False + + if not self.first_time: + await self.reddit_scraper(subreddit_name="CryptoMoonShots") + +
+[docs] + async def reddit_scraper( + self, + limit: int = 15, + subreddit_name: str = "WallStreetBets", + ) -> None: + """ + Scrapes the top reddit posts from the wallstreetbets subreddit and posts them in the wallstreetbets channel. + + Parameters + ---------- + reddit : asyncpraw.Reddit + The reddit instance using the bot's credentials. + limit : int + The number of posts to scrape. + subreddit_name : str + The name of the subreddit to scrape. + + Returns + ------- + None + """ + await update_reddit_ids() + subreddit = await self.reddit.subreddit(subreddit_name) + + counter = 1 + async for submission in subreddit.hot(limit=limit): + if submission.stickied or is_submission_processed(submission.id): + continue + + self.add_id_to_db(submission.id) + + descr = truncate_text(html.unescape(submission.selftext), 4000) + descr = process_description(descr) # Process the description for URLs + + title = truncate_text(html.unescape(submission.title), 250) + img_urls, title = process_submission_media(submission, title) + + embed = create_embed(submission, title, descr, img_urls) + if subreddit_name == "WallStreetBets": + channel = self.wsb_channel + else: + channel = self.cmc_channel + await self.send_embed(embed, img_urls, channel) + + counter += 1 + if counter > 10: + break + + update_db(util.vars.reddit_ids, "reddit_ids")
+ + +
+[docs] + async def send_embed(self, embed: Embed, img_urls: list, channel) -> None: + """ + Send a discord embed, handling multiple images if necessary. + + Parameters + ---------- + embed : discord.Embed + The embed to send. + img_urls : list + The list of image URLs. + channel : discord.TextChannel + The channel to send the embed to. + + Returns + ------- + None + """ + if len(img_urls) > 1: + image_embeds = [embed] + [ + Embed(url=embed.url).set_image(url=img) for img in img_urls[1:10] + ] + webhook = await get_webhook(channel) + await webhook.send( + embeds=image_embeds, + username="FinTwit", + wait=True, + avatar_url=self.bot.user.avatar.url, + ) + else: + await channel.send(embed=embed)
+
+ + + +
+[docs] +async def update_reddit_ids(): + """ + Update the list of reddit IDs, removing those older than 72 hours. + """ + if not util.vars.reddit_ids.empty: + util.vars.reddit_ids = util.vars.reddit_ids.astype( + {"id": str, "timestamp": "datetime64[ns]"} + ) + util.vars.reddit_ids = util.vars.reddit_ids[ + util.vars.reddit_ids["timestamp"] > datetime.now() - timedelta(hours=72) + ]
+ + + +
+[docs] +def is_submission_processed(submission_id: str) -> bool: + """ + Check if a submission has already been processed. + + Parameters + ---------- + submission_id : str + The ID of the submission. + + Returns + ------- + bool + True if the submission has been processed, False otherwise. + """ + if ( + not util.vars.reddit_ids.empty + and submission_id in util.vars.reddit_ids["id"].tolist() + ): + return True + return False
+ + + +
+[docs] +def truncate_text(text: str, max_length: int) -> str: + """ + Truncate text to a maximum length, adding ellipsis if truncated. + + Parameters + ---------- + text : str + The text to truncate. + max_length : int + The maximum length of the text. + + Returns + ------- + str + The truncated text. + """ + if len(text) > max_length: + return text[:max_length] + "..." + return text
+ + + +
+[docs] +def process_submission_media(submission, title: str) -> tuple: + """ + Process the media in a submission, updating the title and extracting image URLs. + + Parameters + ---------- + submission : asyncpraw.models.Submission + The reddit submission. + title : str + The title of the submission. + + Returns + ------- + tuple + A tuple containing the list of image URLs and the updated title. + """ + img_urls = [] + if not submission.is_self: + url = submission.url + if url.endswith((".jpg", ".png", ".gif")): + img_urls.append(url) + title = "🖼️ " + title + elif "gallery" in url: + for image_item in submission.media_metadata.values(): + img_urls.append(image_item["s"]["u"]) + title = "📸🖼️ " + title + elif "v.redd.it" in url: + title = "🎥 " + title + if "images" in submission.preview: + img_urls.append(submission.preview["images"][0]["source"]["url"]) + else: + print("No image found for video post") + return img_urls, title
+ + + +
+[docs] +def create_embed(submission, title: str, descr: str, img_urls: list) -> Embed: + """ + Create a discord embed for a reddit submission. + + Parameters + ---------- + submission : asyncpraw.models.Submission + The reddit submission. + title : str + The title of the submission. + descr : str + The description of the submission. + img_urls : list + The list of image URLs. + + Returns + ------- + discord.Embed + The created embed. + """ + embed = Embed( + title=title, + url="https://www.reddit.com" + submission.permalink, + description=descr, + color=data_sources["reddit"]["color"], + timestamp=datetime.utcfromtimestamp(submission.created_utc), + ) + if img_urls: + embed.set_image(url=img_urls[0]) + embed.set_footer( + text=f"🔼 {submission.score} | 💬 {submission.num_comments} | {submission.link_flair_text}", + icon_url=data_sources["reddit"]["icon"], + ) + return embed
+ + + +
+[docs] +def process_description(description): + """ + Process the description to convert URLs to plain text links unless they are part of a hyperlink with custom text. + + Parameters + ---------- + description : str + The original description text. + + Returns + ------- + str + The processed description. + """ + + # Replace Markdown links with just the URL if the text matches the URL + def replace_markdown_link(match): + text = match.group("text") + url = match.group("url") + if text == url: + return url + return match.group(0) + + description = re.sub(MARKDOWN_LINK_REGEX, replace_markdown_link, description) + + # Replace remaining URLs with just the URL + def replace_url(match): + return match.group("url") + + processed_description = re.sub(URL_REGEX, replace_url, description) + + return processed_description
+ + + +def setup(bot: commands.bot.Bot) -> None: + bot.add_cog(Reddit(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/stock_halts.html b/_modules/cogs/loops/stock_halts.html new file mode 100644 index 00000000..a27151e3 --- /dev/null +++ b/_modules/cogs/loops/stock_halts.html @@ -0,0 +1,589 @@ + + + + + + + + + + cogs.loops.stock_halts — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.stock_halts

+import datetime
+from io import StringIO
+
+# > Discord dependencies
+import discord
+import pandas as pd
+from dateutil import tz
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.afterhours import afterHours
+from util.disc_util import get_channel, get_tagged_users
+
+# Local dependencies
+from util.vars import config, data_sources, post_json_data
+
+
+
+[docs] +class StockHalts(commands.Cog): + """ + This class contains the cog for posting the halted stocks. + It can be configured in the config.yaml file under ["LOOPS"]["STOCK_HALTS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + self.channel = None + self.halt_embed.start() + + @loop(minutes=15) + async def halt_embed(self): + # Dont send if the market is closed + if afterHours(): + return + + if self.channel is None: + self.channel = await get_channel( + self.bot, config["LOOPS"]["STOCK_HALTS"]["CHANNEL"] + ) + + # Get the data + html = await self.get_halt_data() + + if html == {}: + return + + # Remove previous message first + await self.channel.purge(limit=1) + + df = self.format_df(html) + + # Create embed + e = discord.Embed( + title="Halted Stocks", + url="https://www.nasdaqtrader.com/trader.aspx?id=tradehalts", + description="", + color=data_sources["nasdaqtrader"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + # Get the values as string + time = "\n".join(df["Time"].to_list()) + symbol = "\n".join(df["Issue Symbol"].to_list()) + + # Add the values to the embed + e.add_field(name="Time", value=time, inline=True) + e.add_field(name="Symbol", value=symbol, inline=True) + + if "Resumption Time" in df.columns: + resumption = "\n".join(df["Resumption Time"].to_list()) + e.add_field(name="Resumption Time", value=resumption, inline=True) + + e.set_footer( + text="\u200b", + icon_url=data_sources["nasdaqtrader"]["icon"], + ) + + tags = get_tagged_users(df["Issue Symbol"].to_list()) + + await self.channel.send(content=tags, embed=e) + +
+[docs] + def format_df(self, html): + df = pd.read_html(StringIO(html["result"]))[0] + + # Drop NaN columns + df = df.dropna(axis=1, how="all") + + # Drop columns where halt date is not today + df = df[df["Halt Date"] == pd.Timestamp.today().strftime("%m/%d/%Y")] + + # Combine columns into one singular datetime column + df["Time"] = df["Halt Date"] + " " + df["Halt Time"] + df["Time"] = pd.to_datetime(df["Time"], format="%m/%d/%Y %H:%M:%S") + + # Do for resumption as well if the column is not NaN + if "Resumption Date" in df.columns and "Resumption Trade Time" in df.columns: + # Combine columns into one singular datetime column + df["Resumption Time"] = ( + df["Resumption Date"] + " " + df["Resumption Trade Time"] + ) + df["Resumption Time"] = pd.to_datetime( + df["Resumption Time"], format="%m/%d/%Y %H:%M:%S" + ) + + df["Resumption Time"] = ( + df["Resumption Time"] + .dt.tz_localize("US/Eastern") + .dt.tz_convert(tz.tzlocal()) + ) + + df["Resumption Time"] = df["Resumption Time"].dt.strftime("%H:%M:%S") + + # Convert to my own timezone + df["Time"] = df["Time"].dt.tz_localize("US/Eastern").dt.tz_convert(tz.tzlocal()) + + # Convert times to string + df["Time"] = df["Time"].dt.strftime("%H:%M:%S") + + # Replace NaN with ? + df = df.fillna("?") + + # Keep the necessary columns + if "Resumption Time" in df.columns: + df = df[["Time", "Issue Symbol", "Resumption Time"]] + else: + df = df[["Time", "Issue Symbol"]] + + return df
+ + +
+[docs] + async def get_halt_data(self) -> dict: + # Based on https://github.com/reorx/nasdaqtrader-rss/blob/e675af99ace7d384950d6c75144e9efb1d80b5a7/rss/index.py#L18 + headers = { + "Content-Type": "application/json", + "Origin": "https://www.nasdaqtrader.com", + "Referer": "https://www.nasdaqtrader.com/trader.aspx?id=tradehalts", + "Sec-Ch-Ua": '"Not.A/Brand";v="8", "Chromium";v="114", "Google Chrome";v="114"', + "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36", + } + req_data = { + "id": 3, + "method": "BL_TradeHalt.GetTradeHalts", + "params": "[]", + "version": "1.1", + } + + html = await post_json_data( + "https://www.nasdaqtrader.com/RPCHandler.axd", + headers=headers, + json=req_data, + ) + + # Convert to DataFrame + return html
+
+ + + +def setup(bot: commands.Bot) -> None: + """ + This is a necessary method to make the cog loadable. + + Returns + ------- + None + """ + bot.add_cog(StockHalts(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/stocktwits.html b/_modules/cogs/loops/stocktwits.html new file mode 100644 index 00000000..f60b1c9d --- /dev/null +++ b/_modules/cogs/loops/stocktwits.html @@ -0,0 +1,566 @@ + + + + + + + + + + cogs.loops.stocktwits — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.stocktwits

+## > Imports
+# > Standard libraries
+import datetime
+
+# > Discord dependencies
+import discord
+
+# > 3rd party dependencies
+import pandas as pd
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.disc_util import get_channel
+
+# Local dependencies
+from util.vars import config, data_sources, get_json_data
+
+
+
+[docs] +class StockTwits(commands.Cog): + """ + This class contains the cog for posting the most discussed StockTwits tickers. + It can be enabled / disabled in the config under ["LOOPS"]["STOCKTWITS"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + self.channel = None + self.stocktwits.start() + +
+[docs] + async def get_data(self, e: discord.Embed, keyword: str) -> discord.Embed: + """ + Gets the data from StockTwits based on the passed keywords and returns a discord.Embed. + + Parameters + ---------- + e : discord.Embed + The discord.Embed where the data will be added to. + keyword : str + The specific keyword to get the data for. Options are: ts, m_day, wl_ct_day. + + Returns + ------- + discord.Embed + The discord.Embed with the data added to it. + """ + + # Keyword can be "ts", "m_day", "wl_ct_day" + data = await get_json_data( + "https://api.stocktwits.com/api/2/charts/" + keyword, + headers={ + "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36", + }, + ) + + # If no data could be found, return the embed + if data == {}: + return e + + table = pd.DataFrame(data["table"][keyword]) + stocks = pd.DataFrame(data["stocks"]).T + stocks["stock_id"] = stocks.index.astype(int) + full_df = pd.merge(stocks, table, on="stock_id") + full_df.sort_values(by="val", ascending=False, inplace=True) + + # Set types + full_df["price"] = full_df["price"].astype(float).fillna(0) + full_df["change"] = full_df["change"].astype(float).fillna(0) + full_df["symbol"] = full_df["symbol"].astype(str) + full_df["name"] = full_df["name"].astype(str) + + # Format % change + full_df["change"] = full_df["change"].apply( + lambda x: f" (+{round(x,2)}% 📈)" if x > 0 else f" ({round(x,2)}% 📉)" + ) + + # Format price + full_df["price"] = full_df["price"].apply(lambda x: round(x, 3)) + full_df["price"] = full_df["price"].astype(str) + full_df["change"] + + # Show Symbol, Price + change, Score / Count + if keyword == "ts": + name = "Trending" + val = "Score" + elif keyword == "m_day": + name = "Most Active" + val = "Count" + else: + name = "Most Watched" + val = "Count" + + # Set values as string + full_df["val"] = full_df["val"].astype(str) + + # Get the values as string + assets = "\n".join(full_df["symbol"].to_list()) + prices = "\n".join(full_df["price"].to_list()) + values = "\n".join(full_df["val"].to_list()) + + e.add_field(name=name, value=assets, inline=True) + e.add_field(name="Price", value=prices, inline=True) + e.add_field(name=val, value=values, inline=True) + + return e
+ + + @loop(hours=6) + async def stocktwits(self) -> None: + """ + The function posts the StockTwits embeds in the configured channel. + + Returns + ------- + None + """ + if self.channel is None: + self.channel = await get_channel( + self.bot, + config["LOOPS"]["STOCKTWITS"]["CHANNEL"], + config["CATEGORIES"]["STOCKS"], + ) + + e = discord.Embed( + title="StockTwits Rankings", + url="https://stocktwits.com/rankings/trending", + description="", + color=data_sources["stocktwits"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e = await self.get_data(e, "ts") + e = await self.get_data(e, "m_day") + e = await self.get_data(e, "wl_ct_day") + + # Set datetime and icon + e.set_footer( + text="\u200b", + icon_url=data_sources["stocktwits"]["icon"], + ) + + await self.channel.send(embed=e)
+ + + +def setup(bot: commands.bot.Bot) -> None: + bot.add_cog(StockTwits(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/timeline.html b/_modules/cogs/loops/timeline.html new file mode 100644 index 00000000..edb5592a --- /dev/null +++ b/_modules/cogs/loops/timeline.html @@ -0,0 +1,780 @@ + + + + + + + + + + cogs.loops.timeline — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+
+ + + +
+
+ +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.timeline

+from __future__ import annotations
+from typing import List, Optional
+import traceback
+
+import aiohttp
+import discord
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.vars import config
+from util.disc_util import get_channel, get_tagged_users, get_webhook
+from util.tweet_embed import make_tweet_embed
+from util.parse_tweet import parse_tweet
+from util.get_tweet import get_tweet
+
+
+
+[docs] +class Timeline(commands.Cog): + """ + The main Class of this project. This class is responsible for streaming tweets from the Twitter API. + It can be configured in the config.yaml file under ["LOOPS"]["TIMELINE"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + """Initializes the Timeline class. + + Parameters + ---------- + bot : commands.Bot + The bot object from discord.py + """ + self.bot = bot + + charts_channel = config["LOOPS"]["TIMELINE"]["CHARTS_CHANNEL"] + text_channel = config["LOOPS"]["TIMELINE"]["TEXT_CHANNEL"] + + # Set the channels + self.set_channels("STOCKS", charts_channel, text_channel) + self.set_channels("CRYPTO", charts_channel, text_channel) + self.set_channels("FOREX", charts_channel, text_channel) + + # These channels are not crypto or stocks + self.set_channels("IMAGES") + self.set_channels("OTHER") + self.set_channels("NEWS") + + # Get all text channels + self.all_txt_channels.start() + self.get_latest_tweet.start() + +
+[docs] + def set_channels( + self, + name: str, + charts_channel: str = None, + text_channel: str = None, + ) -> None: + """Set channels for each category. + + Parameters + ---------- + name : str + The name of the category. + charts_channel : str + The name of the charts channel. + text_channel : str + The name of the text channel. + """ + if config["LOOPS"]["TIMELINE"][name]["ENABLED"]: + if name in ["STOCKS", "CRYPTO", "FOREX"]: + self.__dict__[f"{name.lower()}_charts_channel"] = get_channel( + self.bot, charts_channel, config["CATEGORIES"][name] + ) + self.__dict__[f"{name.lower()}_text_channel"] = get_channel( + self.bot, text_channel, config["CATEGORIES"][name] + ) + elif name in ["IMAGES", "OTHER"]: + self.__dict__[f"{name.lower()}_channel"] = get_channel( + self.bot, config["LOOPS"]["TIMELINE"][name]["CHANNEL"] + ) + elif name in ["NEWS"]: + self.__dict__[f"{name.lower()}_channel"] = get_channel( + self.bot, + config["LOOPS"]["TIMELINE"][name]["CHANNEL"], + config["CATEGORIES"]["TWITTER"], + ) + + if config["LOOPS"]["TIMELINE"]["NEWS"]["CRYPTO"]["ENABLED"]: + self.crypto_news_channel = get_channel( + self.bot, + config["LOOPS"]["TIMELINE"]["NEWS"]["CHANNEL"], + config["CATEGORIES"]["CRYPTO"], + )
+ + + @loop(hours=1) + async def all_txt_channels(self) -> None: + """Gets all the text channels as Discord object and the names of the channels.""" + self.following_ids = [] + + text_channel_list = [] + text_channel_names = [] + + # Loop over all the text channels + for server in self.bot.guilds: + for channel in server.channels: + if str(channel.type) == "text": + text_channel_list.append(channel) + text_channel_names.append(channel.name.split("┃")[1]) + + # Set the class variables + self.text_channels = text_channel_list + self.text_channel_names = text_channel_names + + @loop(minutes=5) + async def get_latest_tweet(self) -> None: + """Fetches the latest tweets.""" + tweets = await get_tweet() + + # Loop from oldest to newest tweet + for tweet in reversed(tweets): + tweet = tweet["content"] + + # Skip if the tweet is not a timeline item + if tweet["entryType"] != "TimelineTimelineItem": + continue + + await self.on_data(tweet, update_tweet_id=True) + +
+[docs] + async def on_data(self, tweet: dict, update_tweet_id: bool = False) -> None: + """This method is called whenever data is received from the stream. + + Parameters + ---------- + tweet : dict + The raw tweet data. + update_tweet_id : bool, optional + Whether or not to update the tweet ID, by default False + """ + formatted_tweet = parse_tweet(tweet, update_tweet_id=update_tweet_id) + + if formatted_tweet is not None: + ( + text, + user_name, + user_screen_name, + user_img, + tweet_url, + media, + tickers, + hashtags, + e_title, + ) = formatted_tweet + + e, category, base_symbols = await make_tweet_embed( + text, + user_name, + user_img, + tweet_url, + media, + tickers, + hashtags, + e_title, + self.bot, + ) + + # Upload the tweet to the Discord. + await self.upload_tweet(e, category, media, user_screen_name, base_symbols)
+ + +
+[docs] + async def upload_tweet( + self, + e: discord.Embed, + category: Optional[str], + media: List[str], + user_screen_name: str, + tickers: List[str], + ) -> None: + """Uploads tweet in the dedicated Discord channel. + + Parameters + ---------- + e : discord.Embed + The Tweet as a Discord embed object. + category : str, optional + The category of the tweet, used to decide which Discord channel it should be uploaded to. + media : list + The images contained in this tweet. + user_screen_name : str + The user that posted this tweet. + tickers : list + The list of tickers contained in this tweet. + """ + user_channel = None + + # Default channel + channel = self.other_channel + + # Check if there is a user specific channel + if user_screen_name.lower() in self.text_channel_names: + user_channel = self.text_channels[ + self.text_channel_names.index(user_screen_name.lower()) + ] + + # News posters (Do not post news in other channels) + if user_screen_name in config["LOOPS"]["TIMELINE"]["NEWS"]["FOLLOWING"]: + channel = self.news_channel + elif ( + user_screen_name + in config["LOOPS"]["TIMELINE"]["NEWS"]["CRYPTO"]["FOLLOWING"] + ): + channel = self.crypto_news_channel + else: + channel = self.get_channel_based_on_category(category, media) + + await self.post_tweet(channel, e, media, tickers, user_channel, category)
+ + +
+[docs] + def get_channel_based_on_category( + self, category: Optional[str], media: List[str] + ) -> discord.abc.GuildChannel: + """Get the Discord channel based on the category of the tweet. + + Parameters + ---------- + category : str, optional + The category of the tweet. + media : list + The images contained in this tweet. + + Returns + ------- + discord.abc.GuildChannel + The Discord channel. + """ + if category is None: + channel = self.images_channel if media else self.other_channel + else: + channel_type = "charts" if media else "text" + channel = self.__dict__[f"{category}_{channel_type}_channel"] + + return channel
+ + +
+[docs] + async def post_tweet( + self, + channel: discord.abc.GuildChannel, + e: discord.Embed, + media: List[str], + tickers: List[str], + user_channel: Optional[discord.abc.GuildChannel], + category: Optional[str], + ) -> None: + """Formats the tweet and passes it to upload_tweet(). + + Parameters + ---------- + channel : discord.abc.GuildChannel + The Discord channel where the tweet should be posted. + e : discord.Embed + The Tweet as a Discord embed object. + media : list + The images contained in this tweet. + tickers : list + The list of tickers contained in this tweet. + user_channel : discord.abc.GuildChannel, optional + The user-specific Discord channel. + category : str, optional + The category of the tweet. + """ + msgs = [] + + try: + # Create a list of image embeds, max 10 images per post + image_e = [e] + [ + discord.Embed(url=e.url).set_image(url=img) for img in media[1:10] + ] + + # If there are multiple images to be sent, use a webhook to send them all at once + if len(image_e) > 1: + msg = await self.make_and_send_webhook(channel, tickers, image_e) + msgs.append(msg) + + if user_channel: + msg = await self.make_and_send_webhook( + user_channel, tickers, image_e + ) + msgs.append(msg) + + else: + # Use the normal send function + msg = await channel.send(content=get_tagged_users(tickers), embed=e) + msgs.append(msg) + + if user_channel: + msg = await user_channel.send( + content=get_tagged_users(tickers), embed=e + ) + msgs.append(msg) + + # Do this for every message + try: + for msg in msgs: + # Post in highlight channel + await msg.add_reaction("💸") + # Send to user DM + await msg.add_reaction("❤️") + + if category != None: + await msg.add_reaction("🐂") + await msg.add_reaction("🦆") + await msg.add_reaction("🐻") + + except discord.DiscordServerError: + print("Could not add reaction to message") + + except aiohttp.ClientConnectionError: + print("Connection Error posting tweet on timeline") + + except Exception as error: + print("Error posting tweet on timeline", error) + print(traceback.format_exc())
+ + +
+[docs] + async def make_and_send_webhook( + self, + channel: discord.abc.GuildChannel, + tickers: List[str], + image_e: List[discord.Embed], + ) -> discord.Message: + """Creates and sends a webhook. + + Parameters + ---------- + channel : discord.abc.GuildChannel + The Discord channel. + tickers : list + The list of tickers contained in this tweet. + image_e : list + The images contained in this tweet. + + Returns + ------- + discord.Message + The Discord message. + """ + webhook = await get_webhook(channel) + + # Wait so we can use this message as reference + msg = await webhook.send( + content=get_tagged_users(tickers), + embeds=image_e, + username="FinTwit", + wait=True, + avatar_url=self.bot.user.avatar.url, + ) + + return msg
+
+ + + +def setup(bot: commands.Bot) -> None: + """ + This is a necessary method to make the cog loadable. + + Returns + ------- + None + """ + bot.add_cog(Timeline(bot)) +
+ +
+ + + +
+ +
+ +
+
+
+ +
+ +
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/trades.html b/_modules/cogs/loops/trades.html new file mode 100644 index 00000000..79c9771d --- /dev/null +++ b/_modules/cogs/loops/trades.html @@ -0,0 +1,559 @@ + + + + + + + + + + cogs.loops.trades — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.trades

+##> Imports
+import asyncio
+
+import ccxt.pro as ccxt
+
+# > 3rd Party Dependencies
+import pandas as pd
+
+# > Discord dependencies
+from discord.ext import commands
+
+# Local dependencies
+import util.vars
+from util.db import get_db, update_db
+from util.disc_util import get_channel, get_user
+from util.trades_msg import on_msg
+from util.vars import config
+
+
+
+[docs] +class Trades(commands.Cog): + """ + This class contains the cog for posting new trades done by users. + It can be enabled / disabled in the config under ["LOOPS"]["TRADES"]. + """ + + def __init__( + self, bot: commands.Bot, db: pd.DataFrame = get_db("portfolio") + ) -> None: + self.bot = bot + self.trades_channel = None + # Start getting trades + asyncio.create_task(self.trades(db)) + +
+[docs] + async def start_sockets(self, exchange, row, user) -> None: + while True: + try: + msg = await exchange.watchMyTrades() + await on_msg(msg, exchange, self.trades_channel, row, user) + except ccxt.base.errors.AuthenticationError: + # Send message to user and delete from database + print(row) + break + + except Exception as e: + # Maybe do: await exchange.close() and restart the socket + print( + f"Error in trade websocket for {row['user']} and {exchange.id}: ", e + )
+ + +
+[docs] + async def trades(self, db: pd.DataFrame) -> None: + """ + Starts the websockets for each user in the database. + + Parameters + ---------- + db : pd.DataFrame + The database containing all users. + """ + if self.trades_channel is None: + self.trades_channel = await get_channel( + self.bot, config["LOOPS"]["TRADES"]["CHANNEL"] + ) + + tasks = [] + exchanges = [] + + if not db.empty: + # Divide per exchange + binance = db.loc[db["exchange"] == "binance"] + kucoin = db.loc[db["exchange"] == "kucoin"] + + if not binance.empty: + for i, row in binance.iterrows(): + # If using await, it will block other connections + exchange = ccxt.binance( + {"apiKey": row["key"], "secret": row["secret"]} + ) + user = await get_user(self.bot, row["id"]) + + # Make sure that the API keys are valid + try: + exchange.fetch_balance() + except Exception: + # Send message to user and delete from database + await user.send( + "Your Binance API key is invalid, we have removed it from our database." + ) + + # Get the portfolio + util.vars.portfolio_db.drop(i, inplace=True) + + update_db(util.vars.portfolio_db, "portfolio") + + print(f"Removed Binance API key for {row['user']}") + + task = asyncio.create_task(self.start_sockets(exchange, row, user)) + tasks.append(task) + exchanges.append(exchange) + print(f"Started Binance socket for {row['user']}") + + if not kucoin.empty: + for _, row in kucoin.iterrows(): + exchange = ccxt.kucoin( + { + "apiKey": row["key"], + "secret": row["secret"], + "password": row["passphrase"], + } + ) + task = asyncio.create_task( + self.start_sockets( + exchange, row, await get_user(self.bot, row["id"]) + ) + ) + tasks.append(task) + exchanges.append(exchange) + print(f"Started KuCoin socket for {row['user']}") + + # After 24 hours close the exchange and start again + await asyncio.sleep(24 * 60 * 60) + + print("Stopping all sockets") + for task, exchange in zip(tasks, exchanges): + task.cancel() + await exchange.close() + await asyncio.sleep(10) + + # Restart the socket + await self.trades(db)
+
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Trades(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/trending.html b/_modules/cogs/loops/trending.html new file mode 100644 index 00000000..eafeeebd --- /dev/null +++ b/_modules/cogs/loops/trending.html @@ -0,0 +1,773 @@ + + + + + + + + + + cogs.loops.trending — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.trending

+import datetime
+
+# > Discord dependencies
+import discord
+import pandas as pd
+import pytz
+
+# > 3rd party dependencies
+import yahoo_fin.stock_info as si
+from discord.ext import commands
+from discord.ext.tasks import loop
+
+from util.afterhours import afterHours
+from util.cg_data import get_top_categories, get_trending_coins
+from util.disc_util import get_channel
+from util.formatting import (
+    format_change,
+    format_embed,
+    format_embed_length,
+    human_format,
+)
+
+# Local dependencies
+from util.vars import config, data_sources, get_json_data
+
+
+
+
+
+
+def setup(bot: commands.Bot) -> None:
+    bot.add_cog(Trending(bot))
+
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/cogs/loops/yield.html b/_modules/cogs/loops/yield.html new file mode 100644 index 00000000..cd5a515c --- /dev/null +++ b/_modules/cogs/loops/yield.html @@ -0,0 +1,612 @@ + + + + + + + + + + cogs.loops.yield — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for cogs.loops.yield

+# > Standard libraries
+import datetime
+import os
+
+# > Discord dependencies
+import discord
+
+# > 3rd Party Dependencies
+import matplotlib as mpl
+import matplotlib.pyplot as plt
+import numpy as np
+from discord.ext import commands
+from discord.ext.tasks import loop
+from scipy.interpolate import make_interp_spline
+
+from util.disc_util import get_channel
+from util.tv_data import tv
+from util.tv_symbols import EU_bonds, US_bonds
+
+# Local dependencies
+from util.vars import config
+
+
+
+[docs] +class Yield(commands.Cog): + """ + This class contains the cog for posting the US and EU yield curve. + It can be enabled / disabled in the config under ["LOOPS"]["YIELD"]. + """ + + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + self.channel = None + self.post_curve.start() + + @loop(hours=24) + async def post_curve(self) -> None: + """ + Posts the US and EU yield curve in the channel specified in the config. + Charts based on http://www.worldgovernmentbonds.com/country/united-states/ + + Returns + ------- + None + """ + if self.channel is None: + self.channel = await get_channel( + self.bot, config["LOOPS"]["YIELD"]["CHANNEL"] + ) + + plt.style.use("dark_background") # Set the style first + + mpl.rcParams["axes.spines.right"] = False + mpl.rcParams["axes.spines.left"] = False + mpl.rcParams["axes.spines.top"] = False + mpl.rcParams["axes.spines.bottom"] = False + + mpl.rcParams["axes.edgecolor"] = "white" # Set edge color to white + mpl.rcParams["xtick.color"] = "white" # Set x tick color to white + mpl.rcParams["ytick.color"] = "white" # Set y tick color to white + mpl.rcParams["axes.labelcolor"] = "white" # Set label color to white + mpl.rcParams["text.color"] = "white" # Set text color to white + + await self.plot_US_yield() + await self.plot_EU_yield() + + # Add gridlines + plt.grid(axis="y", color="grey", linewidth=0.5, alpha=0.5) + plt.tick_params(axis="y", which="both", left=False) + + frame = plt.gca() + frame.axes.get_xaxis().set_major_formatter(lambda x, _: f"{int(x)}Y") + + frame.axes.set_ylim(0) + frame.axes.get_yaxis().set_major_formatter(lambda x, _: f"{int(x)}%") + + # Set plot parameters + plt.legend(loc="lower center", ncol=2) + plt.xlabel("Residual Maturity") + + # Convert to plot to a temporary image + file_name = "yield.png" + file_path = os.path.join("temp", file_name) + plt.savefig(file_path, bbox_inches="tight", dpi=300) + plt.cla() + plt.close() + + e = discord.Embed( + title="US and EU Yield Curve Rates", + description="", + color=0x000000, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + file = discord.File(file_path, filename=file_name) + e.set_image(url=f"attachment://{file_name}") + + await self.channel.purge(limit=1) + await self.channel.send(file=file, embed=e) + + # Delete yield.png + os.remove(file_path) + +
+[docs] + async def plot_US_yield(self) -> None: + """ + Gets the US yield curve data from TradingView and plots it. + """ + + years = np.array([0.08, 0.15, 0.25, 0.5, 1, 2, 3, 5, 7, 10, 20, 30]) + yield_percentage = await self.get_yield(US_bonds) + + self.make_plot(years, yield_percentage, "c", "US")
+ + +
+[docs] + async def plot_EU_yield(self): + """ + Gets the EU yield curve data from TradingView and plots it. + """ + + years = np.array( + [0.25, 0.5, 0.75, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 15, 20, 25, 30] + ) + yield_percentage = await self.get_yield(EU_bonds) + + self.make_plot(years, yield_percentage, "r", "EU")
+ + +
+[docs] + async def get_yield(self, bonds: list) -> list: + """ + For each bond in the given list, it gets the yield from TradingView. + + Parameters + ---------- + bonds : list + The names of the bonds to get the yield from. + + Returns + ------- + list + The percentages of the yield for each bond. + """ + + yield_percentage = [] + for bond in bonds: + no_exch = bond.split(":")[1] + tv_data = await tv.get_tv_data(no_exch, "forex") + yield_percentage.append(tv_data[0]) + + return yield_percentage
+ + +
+[docs] + def make_plot( + self, years: list, yield_percentage: list, color: str, label: str + ) -> None: + """ + Makes a matplotlib plot of the yield curve. + Each dot is the yield for a specific bond. + Connects a spline through the dots to make a smooth curve. + + Parameters + ---------- + years : list + The years of the yield curve. + yield_percentage : list + The yield percentage for each year. + color : str + The color of the plotted line. + label : str + The label for the plotted line. + """ + + new_X = np.linspace(years.min(), years.max(), 500) + + # Interpolation + spl = make_interp_spline(years, yield_percentage, k=3) + smooth = spl(new_X) + + # Make the plot + plt.rcParams["figure.figsize"] = (10, 5) # Set the figure size + plt.plot(new_X, smooth, color, label=label) + plt.plot(years, yield_percentage, f"{color}o")
+
+ + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(Yield(bot)) +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/discord/ext/tasks.html b/_modules/discord/ext/tasks.html new file mode 100644 index 00000000..6b223c05 --- /dev/null +++ b/_modules/discord/ext/tasks.html @@ -0,0 +1,1214 @@ + + + + + + + + + + discord.ext.tasks — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for discord.ext.tasks

+"""
+The MIT License (MIT)
+
+Copyright (c) 2015-2021 Rapptz
+Copyright (c) 2021-present Pycord Development
+
+Permission is hereby granted, free of charge, to any person obtaining a
+copy of this software and associated documentation files (the "Software"),
+to deal in the Software without restriction, including without limitation
+the rights to use, copy, modify, merge, publish, distribute, sublicense,
+and/or sell copies of the Software, and to permit persons to whom the
+Software is furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in
+all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
+OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
+FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
+DEALINGS IN THE SOFTWARE.
+"""
+
+from __future__ import annotations
+
+import asyncio
+import datetime
+import inspect
+import sys
+import traceback
+from collections.abc import Sequence
+from typing import Any, Awaitable, Callable, Generic, TypeVar, cast
+
+import aiohttp
+
+import discord
+from discord.backoff import ExponentialBackoff
+from discord.utils import MISSING
+
+__all__ = ("loop",)
+
+T = TypeVar("T")
+_func = Callable[..., Awaitable[Any]]
+LF = TypeVar("LF", bound=_func)
+FT = TypeVar("FT", bound=_func)
+ET = TypeVar("ET", bound=Callable[[Any, BaseException], Awaitable[Any]])
+
+
+class SleepHandle:
+    __slots__ = ("future", "loop", "handle")
+
+    def __init__(
+        self, dt: datetime.datetime, *, loop: asyncio.AbstractEventLoop
+    ) -> None:
+        self.loop = loop
+        self.future = future = loop.create_future()
+        relative_delta = discord.utils.compute_timedelta(dt)
+        self.handle = loop.call_later(relative_delta, future.set_result, True)
+
+    def recalculate(self, dt: datetime.datetime) -> None:
+        self.handle.cancel()
+        relative_delta = discord.utils.compute_timedelta(dt)
+        self.handle = self.loop.call_later(relative_delta, self.future.set_result, True)
+
+    def wait(self) -> asyncio.Future[Any]:
+        return self.future
+
+    def done(self) -> bool:
+        return self.future.done()
+
+    def cancel(self) -> None:
+        self.handle.cancel()
+        self.future.cancel()
+
+
+class Loop(Generic[LF]):
+    """A background task helper that abstracts the loop and reconnection logic for you.
+
+    The main interface to create this is through :func:`loop`.
+    """
+
+    def __init__(
+        self,
+        coro: LF,
+        seconds: float,
+        hours: float,
+        minutes: float,
+        time: datetime.time | Sequence[datetime.time],
+        count: int | None,
+        reconnect: bool,
+        loop: asyncio.AbstractEventLoop,
+    ) -> None:
+        self.coro: LF = coro
+        self.reconnect: bool = reconnect
+        self.loop: asyncio.AbstractEventLoop = loop
+        self.count: int | None = count
+        self._current_loop = 0
+        self._handle: SleepHandle = MISSING
+        self._task: asyncio.Task[None] = MISSING
+        self._injected = None
+        self._valid_exception = (
+            OSError,
+            discord.GatewayNotFound,
+            discord.ConnectionClosed,
+            aiohttp.ClientError,
+            asyncio.TimeoutError,
+        )
+
+        self._before_loop = None
+        self._after_loop = None
+        self._before_loop_running = False
+        self._after_loop_running = False
+        self._is_being_cancelled = False
+        self._has_failed = False
+        self._stop_next_iteration = False
+
+        if self.count is not None and self.count <= 0:
+            raise ValueError("count must be greater than 0 or None.")
+
+        self.change_interval(seconds=seconds, minutes=minutes, hours=hours, time=time)
+        self._last_iteration_failed = False
+        self._last_iteration: datetime.datetime = MISSING
+        self._next_iteration = None
+
+        if not inspect.iscoroutinefunction(self.coro):
+            raise TypeError(
+                f"Expected coroutine function, not {type(self.coro).__name__!r}."
+            )
+
+    async def _call_loop_function(self, name: str, *args: Any, **kwargs: Any) -> None:
+        coro = getattr(self, f"_{name}")
+        if coro is None:
+            return
+
+        if name.endswith("_loop"):
+            setattr(self, f"_{name}_running", True)
+
+        if self._injected is not None:
+            await coro(self._injected, *args, **kwargs)
+        else:
+            await coro(*args, **kwargs)
+
+        if name.endswith("_loop"):
+            setattr(self, f"_{name}_running", False)
+
+    def _try_sleep_until(self, dt: datetime.datetime):
+        self._handle = SleepHandle(dt=dt, loop=self.loop)
+        return self._handle.wait()
+
+    async def _loop(self, *args: Any, **kwargs: Any) -> None:
+        backoff = ExponentialBackoff()
+        await self._call_loop_function("before_loop")
+        self._last_iteration_failed = False
+        if self._time is not MISSING:
+            # the time index should be prepared every time the internal loop is started
+            self._prepare_time_index()
+            self._next_iteration = self._get_next_sleep_time()
+        else:
+            self._next_iteration = datetime.datetime.now(datetime.timezone.utc)
+        try:
+            await self._try_sleep_until(self._next_iteration)
+            while True:
+                if not self._last_iteration_failed:
+                    self._last_iteration = self._next_iteration
+                    self._next_iteration = self._get_next_sleep_time()
+                try:
+                    await self.coro(*args, **kwargs)
+                    self._last_iteration_failed = False
+                except self._valid_exception:
+                    self._last_iteration_failed = True
+                    if not self.reconnect:
+                        raise
+                    await asyncio.sleep(backoff.delay())
+                else:
+                    await self._try_sleep_until(self._next_iteration)
+
+                    if self._stop_next_iteration:
+                        return
+
+                    now = datetime.datetime.now(datetime.timezone.utc)
+                    if now > self._next_iteration:
+                        self._next_iteration = now
+                        if self._time is not MISSING:
+                            self._prepare_time_index(now)
+
+                    self._current_loop += 1
+                    if self._current_loop == self.count:
+                        break
+
+        except asyncio.CancelledError:
+            self._is_being_cancelled = True
+            raise
+        except Exception as exc:
+            self._has_failed = True
+            await self._call_loop_function("error", exc)
+            raise exc
+        finally:
+            await self._call_loop_function("after_loop")
+            self._handle.cancel()
+            self._is_being_cancelled = False
+            self._current_loop = 0
+            self._stop_next_iteration = False
+            self._has_failed = False
+
+    def __get__(self, obj: T, objtype: type[T]) -> Loop[LF]:
+        if obj is None:
+            return self
+
+        copy: Loop[LF] = Loop(
+            self.coro,
+            seconds=self._seconds,
+            hours=self._hours,
+            minutes=self._minutes,
+            time=self._time,
+            count=self.count,
+            reconnect=self.reconnect,
+            loop=self.loop,
+        )
+        copy._injected = obj
+        copy._before_loop = self._before_loop
+        copy._after_loop = self._after_loop
+        copy._error = self._error
+        setattr(obj, self.coro.__name__, copy)
+        return copy
+
+    @property
+    def seconds(self) -> float | None:
+        """Read-only value for the number of seconds
+        between each iteration. ``None`` if an explicit ``time`` value was passed instead.
+
+        .. versionadded:: 2.0
+        """
+        if self._seconds is not MISSING:
+            return self._seconds
+
+    @property
+    def minutes(self) -> float | None:
+        """Read-only value for the number of minutes
+        between each iteration. ``None`` if an explicit ``time`` value was passed instead.
+
+        .. versionadded:: 2.0
+        """
+        if self._minutes is not MISSING:
+            return self._minutes
+
+    @property
+    def hours(self) -> float | None:
+        """Read-only value for the number of hours
+        between each iteration. ``None`` if an explicit ``time`` value was passed instead.
+
+        .. versionadded:: 2.0
+        """
+        if self._hours is not MISSING:
+            return self._hours
+
+    @property
+    def time(self) -> list[datetime.time] | None:
+        """Read-only list for the exact times this loop runs at.
+        ``None`` if relative times were passed instead.
+
+        .. versionadded:: 2.0
+        """
+        if self._time is not MISSING:
+            return self._time.copy()
+
+    @property
+    def current_loop(self) -> int:
+        """The current iteration of the loop."""
+        return self._current_loop
+
+    @property
+    def next_iteration(self) -> datetime.datetime | None:
+        """When the next iteration of the loop will occur.
+
+        .. versionadded:: 1.3
+        """
+        if self._task is MISSING:
+            return None
+        elif self._task and self._task.done() or self._stop_next_iteration:
+            return None
+        return self._next_iteration
+
+    async def __call__(self, *args: Any, **kwargs: Any) -> Any:
+        r"""|coro|
+
+        Calls the internal callback that the task holds.
+
+        .. versionadded:: 1.6
+
+        Parameters
+        ------------
+        \*args
+            The arguments to use.
+        \*\*kwargs
+            The keyword arguments to use.
+        """
+
+        if self._injected is not None:
+            args = (self._injected, *args)
+
+        return await self.coro(*args, **kwargs)
+
+    def start(self, *args: Any, **kwargs: Any) -> asyncio.Task[None]:
+        r"""Starts the internal task in the event loop.
+
+        Parameters
+        ------------
+        \*args
+            The arguments to use.
+        \*\*kwargs
+            The keyword arguments to use.
+
+        Raises
+        --------
+        RuntimeError
+            A task has already been launched and is running.
+
+        Returns
+        ---------
+        :class:`asyncio.Task`
+            The task that has been created.
+        """
+
+        if self._task is not MISSING and not self._task.done():
+            raise RuntimeError("Task is already launched and is not completed.")
+
+        if self._injected is not None:
+            args = (self._injected, *args)
+
+        if self.loop is MISSING:
+            self.loop = asyncio.get_event_loop()
+
+        self._task = self.loop.create_task(self._loop(*args, **kwargs))
+        return self._task
+
+    def stop(self) -> None:
+        r"""Gracefully stops the task from running.
+
+        Unlike :meth:`cancel`\, this allows the task to finish its
+        current iteration before gracefully exiting.
+
+        .. note::
+
+            If the internal function raises an error that can be
+            handled before finishing then it will retry until
+            it succeeds.
+
+            If this is undesirable, either remove the error handling
+            before stopping via :meth:`clear_exception_types` or
+            use :meth:`cancel` instead.
+
+        .. versionadded:: 1.2
+        """
+        if self._task is not MISSING and not self._task.done():
+            self._stop_next_iteration = True
+
+    def _can_be_cancelled(self) -> bool:
+        return bool(
+            not self._is_being_cancelled and self._task and not self._task.done()
+        )
+
+    def cancel(self) -> None:
+        """Cancels the internal task, if it is running."""
+        if self._can_be_cancelled():
+            self._task.cancel()
+
+    def restart(self, *args: Any, **kwargs: Any) -> None:
+        r"""A convenience method to restart the internal task.
+
+        .. note::
+
+            Due to the way this function works, the task is not
+            returned like :meth:`start`.
+
+        Parameters
+        ------------
+        \*args
+            The arguments to use.
+        \*\*kwargs
+            The keyword arguments to use.
+        """
+
+        def restart_when_over(
+            fut: Any, *, args: Any = args, kwargs: Any = kwargs
+        ) -> None:
+            self._task.remove_done_callback(restart_when_over)
+            self.start(*args, **kwargs)
+
+        if self._can_be_cancelled():
+            self._task.add_done_callback(restart_when_over)
+            self._task.cancel()
+
+    def add_exception_type(self, *exceptions: type[BaseException]) -> None:
+        r"""Adds exception types to be handled during the reconnect logic.
+
+        By default, the exception types handled are those handled by
+        :meth:`discord.Client.connect`\, which includes a lot of internet disconnection
+        errors.
+
+        This function is useful if you're interacting with a 3rd party library that
+        raises its own set of exceptions.
+
+        Parameters
+        ------------
+        \*exceptions: Type[:class:`BaseException`]
+            An argument list of exception classes to handle.
+
+        Raises
+        --------
+        TypeError
+            An exception passed is either not a class or not inherited from :class:`BaseException`.
+        """
+
+        for exc in exceptions:
+            if not inspect.isclass(exc):
+                raise TypeError(f"{exc!r} must be a class.")
+            if not issubclass(exc, BaseException):
+                raise TypeError(f"{exc!r} must inherit from BaseException.")
+
+        self._valid_exception = (*self._valid_exception, *exceptions)
+
+    def clear_exception_types(self) -> None:
+        """Removes all exception types that are handled.
+
+        .. note::
+
+            This operation obviously cannot be undone!
+        """
+        self._valid_exception = ()
+
+    def remove_exception_type(self, *exceptions: type[BaseException]) -> bool:
+        r"""Removes exception types from being handled during the reconnect logic.
+
+        Parameters
+        ------------
+        \*exceptions: Type[:class:`BaseException`]
+            An argument list of exception classes to handle.
+
+        Returns
+        ---------
+        :class:`bool`
+            Whether all exceptions were successfully removed.
+        """
+        old_length = len(self._valid_exception)
+        self._valid_exception = tuple(
+            x for x in self._valid_exception if x not in exceptions
+        )
+        return len(self._valid_exception) == old_length - len(exceptions)
+
+    def get_task(self) -> asyncio.Task[None] | None:
+        """Fetches the internal task or ``None`` if there isn't one running."""
+        return self._task if self._task is not MISSING else None
+
+    def is_being_cancelled(self) -> bool:
+        """Whether the task is being cancelled."""
+        return self._is_being_cancelled
+
+    def failed(self) -> bool:
+        """Whether the internal task has failed.
+
+        .. versionadded:: 1.2
+        """
+        return self._has_failed
+
+    def is_running(self) -> bool:
+        """Check if the task is currently running.
+
+        .. versionadded:: 1.4
+        """
+        return not bool(self._task.done()) if self._task is not MISSING else False
+
+    async def _error(self, *args: Any) -> None:
+        exception: Exception = args[-1]
+        print(
+            f"Unhandled exception in internal background task {self.coro.__name__!r}.",
+            file=sys.stderr,
+        )
+        traceback.print_exception(
+            type(exception), exception, exception.__traceback__, file=sys.stderr
+        )
+
+    def before_loop(self, coro: FT) -> FT:
+        """A decorator that registers a coroutine to be called before the loop starts running.
+
+        This is useful if you want to wait for some bot state before the loop starts,
+        such as :meth:`discord.Client.wait_until_ready`.
+
+        The coroutine must take no arguments (except ``self`` in a class context).
+
+        Parameters
+        ----------
+        coro: :ref:`coroutine <coroutine>`
+            The coroutine to register before the loop runs.
+
+        Raises
+        ------
+        TypeError
+            The function was not a coroutine.
+        """
+
+        if not inspect.iscoroutinefunction(coro):
+            raise TypeError(
+                f"Expected coroutine function, received {coro.__class__.__name__!r}."
+            )
+
+        self._before_loop = coro
+        return coro
+
+    def after_loop(self, coro: FT) -> FT:
+        """A decorator that register a coroutine to be called after the loop finished running.
+
+        The coroutine must take no arguments (except ``self`` in a class context).
+
+        .. note::
+
+            This coroutine is called even during cancellation. If it is desirable
+            to tell apart whether something was cancelled or not, check to see
+            whether :meth:`is_being_cancelled` is ``True`` or not.
+
+        Parameters
+        ----------
+        coro: :ref:`coroutine <coroutine>`
+            The coroutine to register after the loop finishes.
+
+        Raises
+        ------
+        TypeError
+            The function was not a coroutine.
+        """
+
+        if not inspect.iscoroutinefunction(coro):
+            raise TypeError(
+                f"Expected coroutine function, received {coro.__class__.__name__!r}."
+            )
+
+        self._after_loop = coro
+        return coro
+
+    def error(self, coro: ET) -> ET:
+        """A decorator that registers a coroutine to be called if the task encounters an unhandled exception.
+
+        The coroutine must take only one argument the exception raised (except ``self`` in a class context).
+
+        By default, this prints to :data:`sys.stderr` however it could be
+        overridden to have a different implementation.
+
+        .. versionadded:: 1.4
+
+        Parameters
+        ----------
+        coro: :ref:`coroutine <coroutine>`
+            The coroutine to register in the event of an unhandled exception.
+
+        Raises
+        ------
+        TypeError
+            The function was not a coroutine.
+        """
+        if not inspect.iscoroutinefunction(coro):
+            raise TypeError(
+                f"Expected coroutine function, received {coro.__class__.__name__!r}."
+            )
+
+        self._error = coro  # type: ignore
+        return coro
+
+    def _get_next_sleep_time(self) -> datetime.datetime:
+        if self._sleep is not MISSING:
+            return self._last_iteration + datetime.timedelta(seconds=self._sleep)
+
+        if self._time_index >= len(self._time):
+            self._time_index = 0
+            if self._current_loop == 0:
+                # if we're at the last index on the first iteration, we need to sleep until tomorrow
+                return datetime.datetime.combine(
+                    datetime.datetime.now(self._time[0].tzinfo or datetime.timezone.utc)
+                    + datetime.timedelta(days=1),
+                    self._time[0],
+                )
+
+        next_time = self._time[self._time_index]
+
+        if self._current_loop == 0:
+            self._time_index += 1
+            if (
+                next_time
+                > datetime.datetime.now(
+                    next_time.tzinfo or datetime.timezone.utc
+                ).timetz()
+            ):
+                return datetime.datetime.combine(
+                    datetime.datetime.now(next_time.tzinfo or datetime.timezone.utc),
+                    next_time,
+                )
+            else:
+                return datetime.datetime.combine(
+                    datetime.datetime.now(next_time.tzinfo or datetime.timezone.utc)
+                    + datetime.timedelta(days=1),
+                    next_time,
+                )
+
+        next_date = cast(
+            datetime.datetime, self._last_iteration.astimezone(next_time.tzinfo)
+        )
+        if next_time < next_date.timetz():
+            next_date += datetime.timedelta(days=1)
+
+        self._time_index += 1
+        return datetime.datetime.combine(next_date, next_time)
+
+    def _prepare_time_index(self, now: datetime.datetime = MISSING) -> None:
+        # now kwarg should be a datetime.datetime representing the time "now"
+        # to calculate the next time index from
+
+        # pre-condition: self._time is set
+        time_now = (
+            now
+            if now is not MISSING
+            else datetime.datetime.now(datetime.timezone.utc).replace(microsecond=0)
+        )
+        for idx, time in enumerate(self._time):
+            if time >= time_now.astimezone(time.tzinfo).timetz():
+                self._time_index = idx
+                break
+        else:
+            self._time_index = 0
+
+    def _get_time_parameter(
+        self,
+        time: datetime.time | Sequence[datetime.time],
+        *,
+        dt: type[datetime.time] = datetime.time,
+        utc: datetime.timezone = datetime.timezone.utc,
+    ) -> list[datetime.time]:
+        if isinstance(time, dt):
+            inner = time if time.tzinfo is not None else time.replace(tzinfo=utc)
+            return [inner]
+        if not isinstance(time, Sequence):
+            raise TypeError(
+                "Expected datetime.time or a sequence of datetime.time for ``time``,"
+                f" received {type(time)!r} instead."
+            )
+        if not time:
+            raise ValueError("time parameter must not be an empty sequence.")
+
+        ret: list[datetime.time] = []
+        for index, t in enumerate(time):
+            if not isinstance(t, dt):
+                raise TypeError(
+                    f"Expected a sequence of {dt!r} for ``time``, received"
+                    f" {type(t).__name__!r} at index {index} instead."
+                )
+            ret.append(t if t.tzinfo is not None else t.replace(tzinfo=utc))
+
+        return sorted(set(ret))  # de-dupe and sort times
+
+    def change_interval(
+        self,
+        *,
+        seconds: float = 0,
+        minutes: float = 0,
+        hours: float = 0,
+        time: datetime.time | Sequence[datetime.time] = MISSING,
+    ) -> None:
+        """Changes the interval for the sleep time.
+
+        .. versionadded:: 1.2
+
+        Parameters
+        ----------
+        seconds: :class:`float`
+            The number of seconds between every iteration.
+        minutes: :class:`float`
+            The number of minutes between every iteration.
+        hours: :class:`float`
+            The number of hours between every iteration.
+        time: Union[:class:`datetime.time`, Sequence[:class:`datetime.time`]]
+            The exact times to run this loop at. Either a non-empty list or a single
+            value of :class:`datetime.time` should be passed.
+            This cannot be used in conjunction with the relative time parameters.
+
+            .. versionadded:: 2.0
+
+            .. note::
+
+                Duplicate times will be ignored, and only run once.
+
+        Raises
+        ------
+        ValueError
+            An invalid value was given.
+        TypeError
+            An invalid value for the ``time`` parameter was passed, or the
+            ``time`` parameter was passed in conjunction with relative time parameters.
+        """
+
+        if time is MISSING:
+            seconds = seconds or 0
+            minutes = minutes or 0
+            hours = hours or 0
+            sleep = seconds + (minutes * 60.0) + (hours * 3600.0)
+            if sleep < 0:
+                raise ValueError("Total number of seconds cannot be less than zero.")
+
+            self._sleep = sleep
+            self._seconds = float(seconds)
+            self._hours = float(hours)
+            self._minutes = float(minutes)
+            self._time: list[datetime.time] = MISSING
+        else:
+            if any((seconds, minutes, hours)):
+                raise TypeError("Cannot mix explicit time with relative time")
+            self._time = self._get_time_parameter(time)
+            self._sleep = self._seconds = self._minutes = self._hours = MISSING
+
+        if self.is_running() and not (
+            self._before_loop_running or self._after_loop_running
+        ):
+            if self._time is not MISSING:
+                # prepare the next time index starting from after the last iteration
+                self._prepare_time_index(now=self._last_iteration)
+
+            self._next_iteration = self._get_next_sleep_time()
+            if not self._handle.done():
+                # the loop is sleeping, recalculate based on new interval
+                self._handle.recalculate(self._next_iteration)
+
+
+def loop(
+    *,
+    seconds: float = MISSING,
+    minutes: float = MISSING,
+    hours: float = MISSING,
+    time: datetime.time | Sequence[datetime.time] = MISSING,
+    count: int | None = None,
+    reconnect: bool = True,
+    loop: asyncio.AbstractEventLoop = MISSING,
+) -> Callable[[LF], Loop[LF]]:
+    """A decorator that schedules a task in the background for you with
+    optional reconnect logic. The decorator returns a :class:`Loop`.
+
+    Parameters
+    ----------
+    seconds: :class:`float`
+        The number of seconds between every iteration.
+    minutes: :class:`float`
+        The number of minutes between every iteration.
+    hours: :class:`float`
+        The number of hours between every iteration.
+    time: Union[:class:`datetime.time`, Sequence[:class:`datetime.time`]]
+        The exact times to run this loop at. Either a non-empty list or a single
+        value of :class:`datetime.time` should be passed. Timezones are supported.
+        If no timezone is given for the times, it is assumed to represent UTC time.
+
+        This cannot be used in conjunction with the relative time parameters.
+
+        .. note::
+
+            Duplicate times will be ignored, and only run once.
+
+        .. versionadded:: 2.0
+
+    count: Optional[:class:`int`]
+        The number of loops to do, ``None`` if it should be an
+        infinite loop.
+    reconnect: :class:`bool`
+        Whether to handle errors and restart the task
+        using an exponential back-off algorithm similar to the
+        one used in :meth:`discord.Client.connect`.
+    loop: :class:`asyncio.AbstractEventLoop`
+        The loop to use to register the task, if not given
+        defaults to :func:`asyncio.get_event_loop`.
+
+    Raises
+    ------
+    ValueError
+        An invalid value was given.
+    TypeError
+        The function was not a coroutine, an invalid value for the ``time`` parameter was passed,
+        or ``time`` parameter was passed in conjunction with relative time parameters.
+    """
+
+    def decorator(func: LF) -> Loop[LF]:
+        return Loop[LF](
+            func,
+            seconds=seconds,
+            minutes=minutes,
+            hours=hours,
+            count=count,
+            time=time,
+            reconnect=reconnect,
+            loop=loop,
+        )
+
+    return decorator
+
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/index.html b/_modules/index.html new file mode 100644 index 00000000..404101d0 --- /dev/null +++ b/_modules/index.html @@ -0,0 +1,457 @@ + + + + + + + + + + Overview: module code — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + + + + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/main.html b/_modules/main.html new file mode 100644 index 00000000..e06b0310 --- /dev/null +++ b/_modules/main.html @@ -0,0 +1,543 @@ + + + + + + + + + + main — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for main

+#!/usr/bin/env python3
+# Python 3.8.11
+
+import datetime
+
+##> Imports
+# > Standard library
+import os
+import sys
+
+# Discord libraries
+import discord
+from discord.ext import commands
+from dotenv import load_dotenv
+
+# Load the .env file
+load_dotenv()
+
+from util.disc_util import get_guild, set_emoji
+
+# Import local dependencies
+from util.vars import config
+
+bot = commands.Bot(intents=discord.Intents.all())
+
+
+
+[docs] +@bot.event +async def on_ready() -> None: + """This gets printed on boot up""" + + # Load the loops and listeners + load_folder("loops") + load_folder("listeners") + + guild = get_guild(bot) + print(f"{bot.user} is connected to {guild.name} at {datetime.datetime.now()} \n") + + await set_emoji(guild)
+ + + +
+[docs] +def load_folder(foldername: str) -> None: + """ + Loads all the cogs in the given folder. + Only loads the cogs if the config allows it. + + Parameters + ---------- + foldername: str + The name of the folder to load the cogs from. + + Returns + ------- + None + """ + + # Get enabled cogs + enabled_cogs = [] + + # Check each file in the folder + for file in config[foldername.upper()]: + # Check the contents of the file in the folder + if config[foldername.upper()][file]: + # If the file type is not a boolean, check if it is enabled + if not isinstance(config[foldername.upper()][file], bool): + # Check if the ENABLED key exists + if "ENABLED" in config[foldername.upper()][file]: + # Append if enabled == True + if config[foldername.upper()][file]["ENABLED"]: + enabled_cogs.append(file.lower() + ".py") + else: + # Append the file to enabled cogs, if its value is True + if config[foldername.upper()][file]: + enabled_cogs.append(file.lower() + ".py") + + # Load all cogs + print(f"Loading {foldername} ...") + for filename in os.listdir(f"./src/cogs/{foldername}"): + if filename.endswith(".py") and filename in enabled_cogs: + try: + # Do not start timeline if the -no_timeline argument is given + if filename == "timeline.py" and "-no_timeline" in sys.argv: + continue + + # Overview.py has no setup function, but should be considered as a loop / cog + if filename == "overview.py": + continue + + print("Loading:", filename) + bot.load_extension(f"cogs.{foldername}.{filename[:-3]}") + except discord.ExtensionAlreadyLoaded: + pass + except discord.ExtensionNotFound: + print("Cog not found:", filename) + print()
+ + + +if __name__ == "__main__": + # Start by loading the database + bot.load_extension("util.db") + + # Ensure the all directories exist + os.makedirs("logs", exist_ok=True) + os.makedirs("temp", exist_ok=True) + os.makedirs("data", exist_ok=True) + + # Load commands + load_folder("commands") + + # Read the token from the config + TOKEN = ( + os.getenv("DEBUG_TOKEN") if "-test" in sys.argv else os.getenv("DISCORD_TOKEN") + ) + + if not TOKEN: + print("No Discord token found. Exiting...") + sys.exit(1) + + # Main event loop + bot.run(TOKEN) + # If the bot randomly stops maybe put back old code +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/afterhours.html b/_modules/util/afterhours.html new file mode 100644 index 00000000..784e77b9 --- /dev/null +++ b/_modules/util/afterhours.html @@ -0,0 +1,463 @@ + + + + + + + + + + util.afterhours — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.afterhours

+# Standard libraries
+import datetime
+
+# 3rd party libraries
+from pandas.tseries.holiday import USFederalHolidayCalendar
+
+# Get the public holidays
+cal = USFederalHolidayCalendar()
+us_holidays = cal.holidays(
+    start=datetime.date(datetime.date.today().year, 1, 1).strftime("%Y-%m-%d"),
+    end=datetime.date(datetime.date.today().year, 12, 31).strftime("%Y-%m-%d"),
+).to_pydatetime()
+
+
+
+[docs] +def afterHours() -> bool: + """ + Simple code to check if the current time is after hours in the US. + Source: https://www.reddit.com/r/algotrading/comments/9x9xho/python_code_to_check_if_market_is_open_in_your/ + + Return + ------ + bool + True if it is currently after-hours, False otherwise. + """ + + now = datetime.datetime.now(datetime.timezone(datetime.timedelta(hours=-5), "EST")) + openTime = datetime.time(hour=9, minute=30, second=0) + closeTime = datetime.time(hour=16, minute=0, second=0) + + # If a holiday + if now.strftime("%Y-%m-%d") in us_holidays: + return True + + # If before 0930 or after 1600 + if (now.time() < openTime) or (now.time() > closeTime): + return True + + # If it's a weekend + if now.date().weekday() > 4: + return True + + # Otherwise the market is open + return False
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/cg_data.html b/_modules/util/cg_data.html new file mode 100644 index 00000000..27941b27 --- /dev/null +++ b/_modules/util/cg_data.html @@ -0,0 +1,828 @@ + + + + + + + + + + util.cg_data — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.cg_data

+##> Imports
+# > Standard libaries
+from __future__ import annotations
+
+import numbers
+import os
+import pickle
+import time
+from io import StringIO
+from typing import List, Optional
+
+import pandas as pd
+import tls_client
+from bs4 import BeautifulSoup
+
+# > Third party libraries
+from pycoingecko import CoinGeckoAPI
+
+# Local dependencies
+import util.vars
+from util.formatting import format_change
+from util.tv_data import tv
+from util.vars import stables
+
+cg = CoinGeckoAPI()
+session = tls_client.Session(
+    client_identifier="chrome112", random_tls_extension_order=True
+)
+
+
+
+[docs] +def get_crypto_info(ids): + if len(ids) > 1: + id = None + best_vol = 0 + coin_dict = None + for symbol in ids.values: + # Catch potential errors + try: + coin_info = cg.get_coin_by_id(symbol) + if "usd" in coin_info["market_data"]["total_volume"]: + volume = coin_info["market_data"]["total_volume"]["usd"] + if volume > best_vol: + best_vol = volume + id = symbol + coin_dict = coin_info + except Exception as e: + print("Error getting coin info for", symbol, "Error:", e) + pass + + else: + id = ids.values[0] + # Try in case the CoinGecko API does not work + try: + coin_dict = cg.get_coin_by_id(id) + except Exception as e: + print("Error getting coin info for", id, "Error:", e) + return None, None + + return coin_dict, id
+ + + +
+[docs] +def get_coin_vol(coin_dict: dict) -> float: + if "total_volume" in coin_dict["market_data"].keys(): + if "usd" in coin_dict["market_data"]["total_volume"].keys(): + return coin_dict["market_data"]["total_volume"]["usd"] + else: + return 1
+ + + +
+[docs] +def get_coin_price(coin_dict: dict) -> float: + if "current_price" in coin_dict["market_data"].keys(): + if "usd" in coin_dict["market_data"]["current_price"].keys(): + return coin_dict["market_data"]["current_price"]["usd"] + else: + return 0
+ + + +
+[docs] +def get_coin_exchanges(coin_dict: dict) -> tuple[str, list]: + base = None + exchanges = [] + if "tickers" in coin_dict.keys(): + for info in coin_dict["tickers"]: + if "base" in info.keys(): + # Somtimes the base is a contract instead of ticker + if base is None: + # > 7, because $KOMPETE + if not (info["base"].startswith("0X") or len(info["base"]) > 7): + base = info["base"] + + if "market" in info.keys(): + exchanges.append(info["market"]["name"]) + + return base, exchanges
+ + + +
+[docs] +def get_info_from_dict(coin_dict: dict): + if coin_dict: + if "market_data" in coin_dict.keys(): + volume = get_coin_vol(coin_dict) + price = get_coin_price(coin_dict) + + change = None + if "price_change_percentage_24h" in coin_dict["market_data"].keys(): + if isinstance( + coin_dict["market_data"]["price_change_percentage_24h"], + numbers.Number, + ): + change = round( + coin_dict["market_data"]["price_change_percentage_24h"], 2 + ) + + # Get the exchanges + base, exchanges = get_coin_exchanges(coin_dict) + + return volume, price, change, exchanges, base + return 0, None, None, None, None
+ + + +
+[docs] +async def get_coin_info( + ticker: str, +) -> Optional[tuple[float, str, List[str], float, str, str]]: + """ + Gets the volume, website, exchanges, price, and change of the coin. + This can only be called maximum 50 times per minute. + + Parameters + ---------- + ticker : str + The ticker of the coin. + + Returns + ------- + float + The volume of the coin. + str + The website of the coin. + list[str] + The exchanges of the coin. + float + The price of the coin. + str + The 24h price change of the coin. + str + The base symbol of the coin, e.g. BTC, ETH, etc. + """ + + id = change = None + total_vol = 0 + exchanges = [] + change = "N/A" + + # Remove formatting from ticker input + if ticker not in stables: + for stable in stables: + if ticker.endswith(stable): + ticker = ticker[: -len(stable)] + + # Get the id of the ticker + # Check if the symbol exists + coin_dict = None + if ticker in util.vars.cg_db["symbol"].values: + # Check coin by symbol, i.e. "BTC" + print("Cg_data ticker:") + print(ticker) + coin_dict, id = get_crypto_info( + util.vars.cg_db[util.vars.cg_db["symbol"] == ticker]["id"] + ) + + # Get the information from the dictionary + if coin_dict: + total_vol, price, change, exchanges, base = get_info_from_dict(coin_dict) + + # Try other methods if the information sucks + if total_vol < 50000 or exchanges == [] or change == "N/A": + # As a second options check the TradingView data + price, perc_change, volume, exchange, website = await tv.get_tv_data( + ticker, "crypto" + ) + if volume != 0: + return ( + volume, + website, + exchange, + price, + format_change(perc_change) if perc_change else "N/A", + ticker, + ) + + # Third option is to check by id + elif ticker.lower() in util.vars.cg_db["id"].values: + coin_dict, id = get_crypto_info( + util.vars.cg_db[util.vars.cg_db["id"] == ticker.lower()]["id"] + ) + + # Fourth option is to check by name, i.e. "Bitcoin" + elif ticker in util.vars.cg_db["name"].values: + coin_dict, id = get_crypto_info( + util.vars.cg_db[util.vars.cg_db["name"] == ticker]["id"] + ) + + # Get the information from the dictionary + total_vol, price, change, exchanges, base = get_info_from_dict(coin_dict) + + # remove duplicates and suffix 'exchange' + if exchanges: + exchanges = [x.lower().replace(" exchange", "") for x in exchanges] + exchanges = list(set(exchanges)) + + # Look into this! + if total_vol != 0 and base is None: + print("No base symbol found for:", ticker) + base = ticker + + # Return the information + return ( + total_vol, + ( + f"https://coingecko.com/en/coins/{id}" + if id + else "https://coingecko.com/en/coins/id_not_found" + ), + exchanges, + price, + format_change(change) if change else "N/A", + base, + )
+ + + + + + + +
+[docs] +async def get_top_categories() -> pd.DataFrame | None: + html = session.get("https://www.coingecko.com/en/categories").text + + soup = BeautifulSoup(html, "html.parser") + + table = soup.find("table") + + if table is None: + print("Error getting top categories from CoinGecko, no table found.") + return + + data = [] + for tr in table.find_all("tr")[1:]: + coin_data = {} + + for i, td in enumerate(tr.find_all("td")): + # i == 0 -> rank + + # Category column (including name and link) + if i == 1: + coin_data["Name"] = td.find("a").text + coin_data["Link"] = "https://www.coingecko.com/" + td.find("a")["href"] + + # 24h + if i == 4: + coin_data["24h Change"] = td["data-sort"] + + # Market cap + if i == 6: + coin_data["Market Cap"] = td["data-sort"] + + if i == 7: + coin_data["Volume"] = td["data-sort"] + + if coin_data != {}: + data.append(coin_data) + + return pd.DataFrame(data)
+ + + +
+[docs] +def get_top_vol_coins(length: int = 50) -> list: + CACHE_FILE = "data/top_vol_coins_cache.pkl" + CACHE_EXPIRATION = 24 * 60 * 60 # 24 hours in seconds + # List of symbols to exclude + STABLE_COINS = [ + "OKBUSDT", + "DAIUSDT", + "USDTUSDT", + "USDCUSDT", + "BUSDUSDT", + "TUSDUSDT", + "PAXUSDT", + "EURUSDT", + "GBPUSDT", + "CETHUSDT", + "WBTCUSDT", + ] + + # Check if the cache file exists and is not expired + os.makedirs(CACHE_FILE.split("/")[0], exist_ok=True) + if os.path.exists(CACHE_FILE): + with open(CACHE_FILE, "rb") as f: + cache_data = pickle.load(f) + cache_time = cache_data["timestamp"] + if time.time() - cache_time < CACHE_EXPIRATION: + # Return the cached data if it's not expired + print("Using cached top volume coins") + return cache_data["data"][:length] + + # Fetch fresh data if the cache is missing or expired + df = pd.DataFrame(cg.get_coins_markets("usd"))["symbol"].str.upper() + "USDT" + + sorted_volume = df[~df.isin(STABLE_COINS)] + top_vol_coins = sorted_volume.tolist() + + # Save the result to the cache + with open(CACHE_FILE, "wb") as f: + pickle.dump({"timestamp": time.time(), "data": top_vol_coins}, f) + + return top_vol_coins[:length]
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/confirm_stock.html b/_modules/util/confirm_stock.html new file mode 100644 index 00000000..cbe672b5 --- /dev/null +++ b/_modules/util/confirm_stock.html @@ -0,0 +1,473 @@ + + + + + + + + + + util.confirm_stock — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.confirm_stock

+## > Imports
+# > 3rd Party Dependencies
+import discord
+import yfinance as yf
+from discord.ext import commands
+from discord.ui import Button, View
+
+
+
+[docs] +async def confirm_stock(bot: commands.Bot, ctx: commands.Context, ticker: str) -> bool: + + # Check if this ticker exists + stock_info = yf.Ticker(ticker) + + # If it does not exist let the user know + if stock_info.info["regularMarketPrice"] == None: + + confirm_button = Button( + label="Confirm", + style=discord.ButtonStyle.green, + emoji="✅", + custom_id="confirm", + ) + cancel_button = Button( + label="Cancel", + style=discord.ButtonStyle.red, + emoji="❌", + custom_id="cancel", + ) + + view = View() + view.add_item(confirm_button) + view.add_item(cancel_button) + + # Can also use ctx.followup.send + await ctx.respond( + ( + f"Are you sure {ticker.upper()} is correct? We could not find it on Yahoo Finance.\n" + "Click on \N{WHITE HEAVY CHECK MARK} to continue and on \N{CROSS MARK} to cancel." + ), + view=view, + ) + + res = await bot.wait_for( + "interaction", check=lambda i: i.custom_id == "confirm" + ) + + # If the confirm button was pressed, return True + if res.data["custom_id"] == "confirm": + return True + else: + return False + + return True
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/db.html b/_modules/util/db.html new file mode 100644 index 00000000..c4c391fc --- /dev/null +++ b/_modules/util/db.html @@ -0,0 +1,760 @@ + + + + + + + + + + util.db — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.db

+# > Standard library
+import datetime
+import os
+import sqlite3
+from collections import defaultdict
+
+import numpy as np
+
+# > 3rd party dependencies
+import pandas as pd
+
+# > Discord dependencies
+from discord.ext import commands
+from discord.ext.tasks import loop
+from pycoingecko import CoinGeckoAPI
+from yahoo_fin.stock_info import tickers_nasdaq
+
+# > Local dependencies
+import util.vars
+from util.tv_data import get_tv_ticker_data
+from util.tv_symbols import all_forex_indices, crypto_indices, stock_indices
+
+# Convert emoji to text
+convert_emoji = defaultdict(
+    lambda: "neutral", {"🐻": "bear", "🐂": "bull", "🦆": "neutral"}
+)
+
+
+
+[docs] +class DB(commands.Cog): + def __init__(self, bot: commands.Bot) -> None: + self.bot = bot + + # Check if the data folder exists + os.makedirs("data", exist_ok=True) + + # Start loops + self.set_tv_db.start() + self.set_cg_db.start() + self.set_nasdaq_tickers.start() + + # Set the portfolio and assets db + self.set_portfolio_db() + self.set_assets_db() + self.set_tweets_db() + self.set_reddit_ids_db() + self.set_ideas_ids_db() + self.set_classified_tickers_db() + self.set_options_db() + +
+[docs] + def set_portfolio_db(self): + util.vars.portfolio_db = get_db("portfolio") + if not util.vars.portfolio_db.empty: + util.vars.portfolio_db["id"] = util.vars.portfolio_db["id"].astype(np.int64)
+ + +
+[docs] + def set_assets_db(self): + util.vars.assets_db = get_db("assets") + if not util.vars.assets_db.empty: + util.vars.assets_db["id"] = util.vars.assets_db["id"].astype(np.int64)
+ + +
+[docs] + def set_tweets_db(self): + util.vars.tweets_db = get_db("tweets")
+ + +
+[docs] + def set_options_db(self): + util.vars.options_db = get_db("options")
+ + +
+[docs] + def set_reddit_ids_db(self): + util.vars.reddit_ids = get_db("reddit_ids")
+ + +
+[docs] + def set_ideas_ids_db(self): + util.vars.ideas_ids = get_db("ideas_ids")
+ + +
+[docs] + def set_classified_tickers_db(self): + util.vars.classified_tickers = get_db("classified_tickers")
+ + + @loop(hours=24) + async def set_nasdaq_tickers(self): + try: + util.vars.nasdaq_tickers = tickers_nasdaq() + update_db(pd.DataFrame(util.vars.nasdaq_tickers), "nasdaq_tickers") + + except Exception as e: + print("Failed to get new nasdaq tickers, error:", e) + nasdaq_tickers = get_db("nasdaq_tickers") + # Convert the dataframe to list + util.vars.nasdaq_tickers = nasdaq_tickers.iloc[:, 0].tolist() + + # Set the important database variables on startup and refresh every 24 hours + @loop(hours=24) + async def set_cg_db(self): + # Saves all CoinGecko coins, maybe refresh this daily + cg = CoinGeckoAPI() + cg_coins = pd.DataFrame(cg.get_coins_list()) + cg_coins["symbol"] = cg_coins["symbol"].str.upper() + + # Save cg_coins to database + update_db(cg_coins, "cg_coins") + + # Set cg_coins + util.vars.cg_db = cg_coins + + @loop(hours=24) + async def set_tv_db(self): + """ + Gets the data from TradingView and saves it to the database. + """ + + # In case the function below fails + util.vars.stocks = get_db("tv_stocks") + util.vars.crypto = get_db("tv_crypto") + util.vars.forex = get_db("tv_forex") + util.vars.cfd = get_db("tv_cfd") + + # Get the current symbols and exchanges on TradingView + tv_stocks = await get_tv_ticker_data( + "https://scanner.tradingview.com/america/scan", stock_indices + ) + tv_crypto = await get_tv_ticker_data( + "https://scanner.tradingview.com/crypto/scan", crypto_indices + ) + tv_forex = await get_tv_ticker_data( + "https://scanner.tradingview.com/forex/scan", all_forex_indices + ) + + # tv_cfd = await get_tv_ticker_data("https://scanner.tradingview.com/cfd/scan") + + # Save the data to the database + for db, name in [ + (tv_stocks, "tv_stocks"), + (tv_crypto, "tv_crypto"), + (tv_forex, "tv_forex"), + # (tv_cfd, "tv_cfd"), + ]: + if not db.empty: + update_db(db, name) + + if name == "tv_stocks": + util.vars.stocks = db + elif name == "tv_crypto": + util.vars.crypto = db + elif name == "tv_forex": + util.vars.forex = db
+ + # elif name == "tv_cfd": + # util.vars.cfd = db + + +def setup(bot: commands.Bot) -> None: + bot.add_cog(DB(bot)) + + +
+[docs] +def remove_old_rows(db: pd.DataFrame, days: int) -> pd.DataFrame: + """ + Removes the old rows from the database and return it. + """ + + # Set timestamp column to datetime + db["timestamp"] = pd.to_datetime(db["timestamp"]) + + return db[db["timestamp"] > datetime.datetime.now() - datetime.timedelta(days=days)]
+ + + +
+[docs] +def merge_and_update( + main_db: pd.DataFrame, new_data: pd.DataFrame, db_name: str +) -> pd.DataFrame: + merged = pd.concat([main_db, new_data], ignore_index=True) + update_db(merged, db_name) + return merged
+ + + +
+[docs] +def clean_old_db(db, days: int = 1) -> pd.DataFrame: + """ + Cleans the tweets database and returns it. + + Returns + ------- + pd.Dataframe + The cleaned tweets database. + """ + + # If the database is empty, do nothing and return + if db.empty: + return db + + # Set the types + try: + db = remove_old_rows(db, days) + return db + except Exception as e: + print("Error in clean_old_db:", e) + print(db.to_string())
+ + + +
+[docs] +def update_tweet_db( + tickers: list, user: str, sentiment: str, categories: list, changes: list +) -> None: + """ + Updates thet tweet database variable using the info provided. + + Parameters + ---------- + tickers : list + The list of tickers. + user : str + The name of the user. + sentiment : str + The sentiment of the tweet. + categories : list + The categories of the tickers. + """ + + # Prepare new data + dict_list = [] + + for i in range(len(tickers)): + # Remove emoji at end + change = changes[i] + if change: + if "%" in change: + change = change[:-1] + else: + change = "None" + else: + change = "None" + + dict_list.append( + { + "ticker": tickers[i], + "user": user, + "sentiment": convert_emoji[sentiment], + "category": categories[i], + "change": change, + } + ) + + # Convert it to a dataframe + tweet_db = pd.DataFrame(dict_list) + + # Add current time + tweet_db["timestamp"] = datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S") + + util.vars.tweets_db = clean_old_db(util.vars.tweets_db, 1) + util.vars.tweets_db = merge_and_update(util.vars.tweets_db, tweet_db, "tweets")
+ + + +
+[docs] +def get_db(database_name: str) -> pd.DataFrame: + """ + Get the database saved under data/<database_name>.pkl. + If it does not exist return an empty dataframe. + + Parameters + ---------- + str + Name of the database to get. + + Returns + ------- + pd.DataFrame + Database saved under data/<database_name>.pkl. + """ + + script_dir = os.path.dirname(__file__) + db_loc = os.path.join(script_dir, "..", "..", "data", f"{database_name}.db") + + try: + cnx = sqlite3.connect(db_loc) + return pd.read_sql_query(f"SELECT * FROM {database_name}", cnx) + except Exception: + print(f"No {database_name}.db found, returning empty db") + return pd.DataFrame()
+ + + +
+[docs] +def update_db(db: pd.DataFrame, database_name: str) -> None: + """ + Update the database saved under data/database_name.pkl using db as the new database. + + Parameters + ---------- + pd.DatFrame + Database to use for updating old database. + str + Name of the database to update. + + Returns + ------- + None + """ + + db_loc = f"data/{database_name}.db" + + # Convert everything to string to prevent errors + # Using map on each column + for column in db.columns: + db[column] = db[column].map(str) + + try: + db.to_sql( + database_name, sqlite3.connect(db_loc), if_exists="replace", index=False + ) + except Exception as e: + print( + f"Error updating {database_name}.db: {e}.\nTried to update database:\n{db.to_string()}" + )
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/disc_util.html b/_modules/util/disc_util.html new file mode 100644 index 00000000..daada58a --- /dev/null +++ b/_modules/util/disc_util.html @@ -0,0 +1,609 @@ + + + + + + + + + + util.disc_util — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.disc_util

+from typing import Optional
+
+import discord
+from discord.ext import commands
+
+import util.vars
+from util.vars import guild_name
+
+
+
+[docs] +def get_guild(bot: commands.Bot) -> discord.Guild: + """ + Returns the guild / server the bot is currently connected to. + + Parameters + ---------- + commands.Bot + The bot object. + + Returns + ------- + discord.Guild + The guild / server the bot is currently connected to. + """ + + return discord.utils.get( + bot.guilds, + # Return the debug server if -test is used as an argument + name=guild_name, + )
+ + + +
+[docs] +async def get_channel( + bot: commands.Bot, channel_name: str, category_name: str = None +) -> discord.TextChannel: + """ + Returns the discord.TextChannel object of the channel with the given name. + + Parameters + ---------- + bot : commands.Bot + The bot object. + channel_name : str + The name of the channel. + + Returns + ------- + discord.TextChannel + The discord.TextChannel object of the channel with the given name. + """ + + for guild in bot.guilds: + if guild.name == guild_name: + for channel in guild.channels: + if channel.name == channel_name: + if category_name is None: + return channel + else: + if channel.category: + if channel.category.name == category_name: + return channel + + print( + f"Channel named: {channel_name}, with category {category_name} not found in guild: {guild_name}.\nCreating it..." + ) + + # If the channel is not found, create it + if category_name: + category = discord.utils.get(guild.categories, name=category_name) + channel = await guild.create_text_channel(channel_name, category=category) + + # Maybe read the category from the config file + channel = await guild.create_text_channel(channel_name) + return channel
+ + + +
+[docs] +async def set_emoji(guild) -> dict: + """ + Returns the custom emoji with the given name. + + Parameters + ---------- + bot : commands.Bot + The bot object. + emoji : str + The name of the emoji. + + Returns + ------- + discord.Emoji + The custom emoji with the given name. + """ + # https://docs.pycord.dev/en/stable/api.html?highlight=emojis#discord.on_guild_emojis_update + # Could use this event to update the emojis if they change + + emojis = await guild.fetch_emojis() + + for emoji in emojis: + util.vars.custom_emojis[emoji.name] = emoji
+ + + +
+[docs] +async def get_user(bot: commands.Bot, user_id: int) -> discord.User: + """ + Gets the discord.User object of the user with the given id. + + Parameters + ---------- + bot : commands.Bot + The bot object. + user_id : int + The id of the user. + + Returns + ------- + discord.User + The discord.User object of the user with the given id. + """ + + return await bot.fetch_user(user_id)
+ + + +
+[docs] +def get_tagged_users(tickers: list) -> Optional[str]: + """ + Tags the users with the tickers in their portfolio that are mentioned in the message. + + Parameters + ---------- + tickers : list + The list of tickers mentioned in the message. + + Returns + ------- + Optional[str] + The message of the users that need to be tagged. + """ + + # Get the stored db + if not util.vars.assets_db.empty: + matching_users = ( + util.vars.assets_db[util.vars.assets_db["asset"].isin(tickers)]["id"] + .dropna() + .tolist() + ) + unique_users = list(set(matching_users)) + + if unique_users: + # Make it one message for all the users + return " ".join([f"<@!{user}>" for user in unique_users])
+ + + +
+[docs] +async def get_webhook(channel: discord.TextChannel) -> discord.Webhook: + """ + Checks if there is a webhook in the given channel and returns it. + If there is not a webhook for a channel, then it creates one. + + Parameters + ---------- + channel : discord.TextChannel + The channel to check for a webhook. + + Returns + ------- + discord.Webhook + The webhook for the given channel. + """ + + webhook = await channel.webhooks() + + if not webhook: + webhook = await channel.create_webhook(name=channel.name) + print(f"Created webhook for {channel.name}") + else: + webhook = webhook[0] + + return webhook
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/earnings_scraper.html b/_modules/util/earnings_scraper.html new file mode 100644 index 00000000..3e0fa327 --- /dev/null +++ b/_modules/util/earnings_scraper.html @@ -0,0 +1,475 @@ + + + + + + + + + + util.earnings_scraper — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+
+ + + +
+
+ +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.earnings_scraper

+## > Imports
+# Standard imports
+import json
+import requests
+import time
+
+
+
+[docs] +class YahooEarningsCalendar: + """ + This is the class for fetching earnings data from Yahoo! Finance, built by https://github.com/wenboyu2. + """ + + def _get_data_dict(self, url: str) -> dict: + + # Sleep 60*60 / 2000 = 1.8 seconds to prevent rate limit + time.sleep(1.8) + page = requests.get( + url, + headers={ + "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36" + }, + ) + page_content = page.content.decode(encoding="utf-8", errors="strict") + page_data_string = [ + row + for row in page_content.split("\n") + if row.startswith("root.App.main = ") + ][0][:-1] + page_data_string = page_data_string.split("root.App.main = ", 1)[1] + + return json.loads(page_data_string) + +
+[docs] + def get_next_earnings_date(self, symbol: str): + """Gets the next earnings date of symbol + Args: + symbol: A ticker symbol + Returns: + Unix timestamp of the next earnings date + Raises: + Exception: When symbol is invalid or earnings date is not available + """ + url = f"https://finance.yahoo.com/quote/{symbol}" + + try: + page_data_dict = self._get_data_dict(url) + return page_data_dict["context"]["dispatcher"]["stores"][ + "QuoteSummaryStore" + ]["calendarEvents"]["earnings"]["earningsDate"][0]["raw"] + except: + raise Exception("Invalid Symbol or Unavailable Earnings Date")
+ + +
+[docs] + def get_earnings_of(self, symbol: str) -> list: + """Returns all the earnings dates of a symbol + Args: + symbol: A ticker symbol + Returns: + Array of all earnings dates with supplemental information + Raises: + Exception: When symbol is invalid or earnings date is not available + """ + url = f"https://finance.yahoo.com/calendar/earnings?symbol={symbol}" + + try: + page_data_dict = self._get_data_dict(url) + return page_data_dict["context"]["dispatcher"]["stores"][ + "ScreenerResultsStore" + ]["results"]["rows"] + except: + raise Exception("Invalid Symbol or Unavailable Earnings Date")
+
+ +
+ +
+ + + +
+ +
+ +
+
+
+ +
+ +
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/exchange_data.html b/_modules/util/exchange_data.html new file mode 100644 index 00000000..8cf38cee --- /dev/null +++ b/_modules/util/exchange_data.html @@ -0,0 +1,589 @@ + + + + + + + + + + util.exchange_data — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.exchange_data

+import traceback
+
+import ccxt.async_support as ccxt
+import numpy as np
+import pandas as pd
+
+from util.vars import stables
+
+
+
+[docs] +async def get_data(row) -> pd.DataFrame: + exchange_info = {"apiKey": row["key"], "secret": row["secret"]} + + if row["exchange"] == "binance": + exchange = ccxt.binance(exchange_info) + exchange.options["recvWindow"] = 60000 + elif row["exchange"] == "kucoin": + exchange_info["password"] = row["passphrase"] + exchange = ccxt.kucoin(exchange_info) + + try: + balances = await get_balance(exchange) + + if balances == "invalid API key": + await exchange.close() + return "invalid API key" + + # Create a list of dictionaries + owned = [] + + for symbol, amount in balances.items(): + usd_val, percentage = await get_usd_price(exchange, symbol) + worth = amount * usd_val + + # Add price change + + if worth < 5: + continue + + buying_price = await get_buying_price(exchange, symbol) + + # If buying price is 0 then it is not known what the price was + owned.append( + { + "asset": symbol, + "buying_price": buying_price, + "owned": amount, + "exchange": exchange.id, + "id": row["id"], + "user": row["user"], + "worth": round(worth, 2), + "price": usd_val, + "change": percentage, + } + ) + + df = pd.DataFrame(owned) + + # Se tthe types + if not df.empty: + df = df.astype( + { + "asset": str, + "buying_price": float, + "owned": float, + "exchange": str, + "id": np.int64, + "user": str, + "worth": float, + "price": float, + "change": float, + } + ) + + await exchange.close() + return df + except Exception as e: + await exchange.close() + print("Error in get_data(). Error:", e) + print(traceback.format_exc())
+ + + +
+[docs] +async def get_balance(exchange) -> dict: + try: + balances = await exchange.fetchBalance() + total_balance = balances["total"] + if total_balance is None: + return "invalid API key" + return {k: v for k, v in total_balance.items() if v > 0} + except Exception: + return {}
+ + + +
+[docs] +async def get_usd_price(exchange, symbol: str) -> tuple[float, float]: + """ + Returns the price of the symbol in USD. + Symbol must be in the format 'BTC/USDT'. + """ + # Directly return for USDT or when symbol is a known stable coin + if symbol == "USDT" or symbol in stables: + return 1.0, 0.0 + + # Helper function to fetch price and change + async def fetch_price(symbol_pair: str): + try: + price = await exchange.fetchTicker(symbol_pair) + exchange_price = price.get("last", 0) + if exchange_price is None: + exchange_price = 0 + exchange_price = float(exchange_price) + exchange_change = price.get("percentage", 0) + if exchange_change is None: + exchange_change = 0 + exchange_change = float(exchange_change) + return exchange_price, exchange_change + except (ccxt.BadSymbol, ccxt.RequestTimeout): + return None # Use None to indicate a failed fetch + except ccxt.ExchangeError as e: + print(f"Exchange error for {symbol_pair} on {exchange.id}: {e}") + return None + except Exception as e: + print(f"Error fetching {symbol_pair} on {exchange.id}: {e}") + return None + + # Attempt to fetch price for each stable coin pairing + for usd in stables: + result = await fetch_price(f"{symbol}/{usd}") + if result: + return result + + # Fallback if no price found for any stable pairing + return 0.0, 0.0
+ + + +
+[docs] +async def get_buying_price(exchange, symbol, full_sym: bool = False) -> float: + # Maybe try different quote currencies when returned list is empty + if symbol in stables: + return 1 + + symbol = symbol + "/USDT" if not full_sym else symbol + + params = {} + if exchange.id == "kucoin": + params = {"side": "buy"} + try: + trades = await exchange.fetchClosedOrders(symbol, params=params) + except ccxt.BadSymbol: + return 0 + except ccxt.RequestTimeout: + return 0 + if type(trades) == list: + if len(trades) > 0: + if exchange.id == "binance": + # Filter list for side:buy + trades = [trade for trade in trades if trade["info"]["side"] == "BUY"] + if len(trades) == 0: + return 0 + + return float(trades[-1]["price"]) + + return 0
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/formatting.html b/_modules/util/formatting.html new file mode 100644 index 00000000..9351a319 --- /dev/null +++ b/_modules/util/formatting.html @@ -0,0 +1,674 @@ + + + + + + + + + + util.formatting — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.formatting

+# Standard libaries
+import datetime
+from math import floor, log
+
+import discord
+
+# Third party libraries
+import pandas as pd
+
+from util.vars import data_sources
+
+
+
+[docs] +def format_change(change: float) -> str: + """ + Converts a float to a string with a plus sign if the float is positive, and a minus sign if the float is negative. + + Parameters + ---------- + change : float + The percentual change of an asset. + + Returns + ------- + str + The formatted change. + """ + if change is None: + return "N/A" + + if isinstance(change, str): + # Try to convert to float + try: + change = float(change) + except ValueError: + change = 0 + + # Round to 2 decimals + change = round(change, 2) + + return f"+{change}% 📈" if change > 0 else f"{change}% 📉"
+ + + +
+[docs] +def human_format(number: float, absolute: bool = False, decimals: int = 0) -> str: + """ + Takes a number and returns a human readable string. + Taken from: https://stackoverflow.com/questions/579310/formatting-long-numbers-as-strings-in-python/45846841. + + Parameters + ---------- + number : float + The number to be formatted. + absolute : bool + If True, the number will be converted to its absolute value. + decimals : int + The number of decimals to be used. + + Returns + ------- + str + The formatted number as a string. + """ + + # Try to convert to float + if isinstance(number, str): + try: + number = float(number) + except ValueError: + number = 0 + + if number == 0: + return "0" + + # https://idlechampions.fandom.com/wiki/Large_number_abbreviations + units = ["", "K", "M", "B", "t", "q"] + k = 1000.0 + magnitude = int(floor(log(abs(number), k))) + + if decimals > 0: + rounded_number = round(number / k**magnitude, decimals) + else: + rounded_number = int(number / k**magnitude) + + if absolute: + rounded_number = abs(rounded_number) + + return f"{rounded_number}{units[magnitude]}"
+ + + +
+[docs] +def format_embed_length(data: list) -> list: + """ + If the length of the data is greater than 1024 characters, it will be shortened to that amount. + + Parameters + ---------- + data : list + The list containing the description for an embed. + + Returns + ------- + list + The shortened description. + """ + + for x in range(len(data)): + if len(data[x]) > 1024: + data[x] = data[x][:1024].split("\n")[:-1] + # Fix everything that is not x + for y in range(len(data)): + if x != y: + data[y] = "\n".join(data[y].split("\n")[: len(data[x])]) + + data[x] = "\n".join(data[x]) + + return data
+ + + +# Used in gainers, losers loops +
+[docs] +async def format_embed(og_df: pd.DataFrame, type: str, source: str) -> discord.Embed: + """ + Formats the dataframe to an embed. + + Parameters + ---------- + df : pd.DataFrame + A dataframe with the columns: + Symbol + Price + % Change + Volume + type : str + The type used in the title of the embed + source : str + The source used for this data + + Returns + ------- + discord.Embed + A Discord embed containing the formatted data + """ + + df = og_df.copy() + + if source == "binance": + url = "https://www.binance.com/en/altcoins/gainers-losers" + color = data_sources["binance"]["color"] + icon_url = data_sources["binance"]["icon"] + name = "Coin" + elif source == "yahoo": + url = "https://finance.yahoo.com/most-active" + color = data_sources["yahoo"]["color"] + icon_url = data_sources["yahoo"]["icon"] + name = "Stock" + elif source == "coingecko": + url = "https://www.coingecko.com/en/watchlists/trending-crypto" + color = data_sources["coingecko"]["color"] + icon_url = data_sources["coingecko"]["icon"] + name = "Coin" + elif source == "coinmarketcap": + url = "https://coinmarketcap.com/trending-cryptocurrencies/" + color = data_sources["coinmarketcap"]["color"] + icon_url = data_sources["coinmarketcap"]["icon"] + name = "Coin" + elif source.startswith("tradingview"): + color = data_sources["tradingview"]["color"] + icon_url = data_sources["tradingview"]["icon"] + name = "Stock" + if source == "tradingview-premarket": + url = "https://www.tradingview.com/markets/stocks-usa/market-movers-active-pre-market-stocks/" + elif source == "tradingview-afterhours": + url = "https://www.tradingview.com/markets/stocks-usa/market-movers-active-after-hours-stocks/" + + e = discord.Embed( + title=f"Top {len(df)} {type}", + url=url, + description="", + color=color, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + if source == "yahoo": + # Format the data + df.rename(columns={"Price (Intraday)": "Price"}, inplace=True) + + # Add website to symbol + df["Symbol"] = ( + "[" + + df["Symbol"] + + "](https://finance.yahoo.com/quote/" + + df["Symbol"] + + ")" + ) + + # Only these columns are necessary + df = df[["Symbol", "Price", "% Change", "Volume"]] + + if not source.startswith("tradingview"): + df = df.astype( + {"Symbol": str, "Price": float, "% Change": float, "Volume": float} + ) + df = df.round({"Price": 3, "% Change": 2, "Volume": 0}) + else: + df = df.astype( + {"Symbol": str, "Price": float, "% Change": float, "Volume": str} + ) + df = df.round({"Price": 3, "% Change": 2}) + + # Apply format_change + df["% Change"] = df["% Change"].apply(format_change) + + # Post symbol, current price (weightedAvgPrice) + change, volume + df["Price"] = "$" + df["Price"].astype(str) + " (" + df["% Change"] + ")" + + # Format volume if it is not done already + if not source.startswith("tradingview"): + df["Volume"] = df["Volume"].apply(lambda x: "$" + human_format(x)) + + ticker = "\n".join(df["Symbol"].tolist()) + prices = "\n".join(df["Price"].tolist()) + vol = "\n".join(df["Volume"].astype(str).tolist()) + + # Prevent possible overflow + ticker, prices, vol = format_embed_length([ticker, prices, vol]) + + e.add_field( + name=name, + value=ticker, + inline=True, + ) + + e.add_field( + name="Price", + value=prices, + inline=True, + ) + + e.add_field( + name="Volume", + value=vol, + inline=True, + ) + + # Set empty text as footer, so we can see the icon + e.set_footer(text="\u200b", icon_url=icon_url) + + return e
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/get_tweet.html b/_modules/util/get_tweet.html new file mode 100644 index 00000000..e88fa4f7 --- /dev/null +++ b/_modules/util/get_tweet.html @@ -0,0 +1,473 @@ + + + + + + + + + + util.get_tweet — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.get_tweet

+import json
+
+import uncurl
+
+from util.vars import get_json_data
+
+# Read curl.txt
+try:
+    with open("curl.txt", "r", encoding="utf-8") as file:
+        cURL = uncurl.parse_context("".join([line.strip() for line in file]))
+except Exception as e:
+    cURL = None
+    print("Error: Could not read curl.txt:", e)
+
+
+
+[docs] +async def get_tweet(): + if cURL is None: + print("Error: no curl.txt file found. Timelines will not be updated.") + return [] + result = await get_json_data( + cURL.url, + headers=dict(cURL.headers), + cookies=dict(cURL.cookies), + json_data=json.loads(cURL.data), + text=False, + ) + + if result == {}: + return [] + + # TODO: Ignore x-premium alerts + if "data" in result: + if "home" in result["data"]: + if "home_timeline_urt" in result["data"]["home"]: + if "instructions" in result["data"]["home"]["home_timeline_urt"]: + if ( + "entries" + in result["data"]["home"]["home_timeline_urt"]["instructions"][ + 0 + ] + ): + return result["data"]["home"]["home_timeline_urt"][ + "instructions" + ][0]["entries"] + + try: + result["data"]["home"]["home_timeline_urt"]["instructions"][0]["entries"] + except Exception as e: + print("Error in get_tweet():", e) + with open("logs/get_tweet_error.json", "w") as f: + json.dump(result, f, indent=4) + + return []
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/parse_tweet.html b/_modules/util/parse_tweet.html new file mode 100644 index 00000000..3bc1e5f4 --- /dev/null +++ b/_modules/util/parse_tweet.html @@ -0,0 +1,678 @@ + + + + + + + + + + util.parse_tweet — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.parse_tweet

+import datetime
+import json
+import re
+from typing import List
+
+# > Local imports
+import util.vars
+
+
+
+[docs] +def remove_twitter_url_at_end(text: str) -> str: + """ + Removes a t.co URL at the end of a text string. + + Parameters + ---------- + text : str + The text from which to remove the URL. + + Returns + ------- + str + The text with the URL removed. + """ + pattern = r"(https?://t\.co/\S+)$" + return re.sub(pattern, "", text)
+ + + +
+[docs] +def get_user_info(tweet: dict, key: str) -> str: + return tweet["core"]["user_results"]["result"]["legacy"][key]
+ + + +
+[docs] +def get_entities(tweet: dict, key: str) -> List[str]: + """ + Retrieves entities from a tweet. + + Parameters + ---------- + tweet : dict + The tweet from which to retrieve entities. + key : str + The key of the entities to retrieve. + + Returns + ------- + List[str] + The retrieved entities, or an empty list if the key does not exist. + """ + if "legacy" in tweet: + if "entities" in tweet["legacy"]: + entities = tweet["legacy"]["entities"].get(key) + return [entity["text"] for entity in entities] if entities else [] + + print("Tweet contains no entities") + return []
+ + + +
+[docs] +def save_errored_tweet(tweet, error_msg: str): + print(error_msg) + # Get current time as a string for the filename + current_time = datetime.datetime.now().strftime("%Y-%m-%d_%H-%M-%S") + + # Write tweet content to a JSON file in the logs directory + with open(f"logs/error_tweet_{current_time}.json", "w", encoding="utf-8") as file: + json.dump(tweet, file, ensure_ascii=False, indent=4)
+ + + +
+[docs] +def parse_tweet(tweet: dict, update_tweet_id: bool = False): + reply = None + is_long_tweet = False + + ## TODO: split the below logic up into functions + + # To be able to get the tweet and the reply + if "items" in tweet.keys(): + reply = tweet["items"][1]["item"]["itemContent"]["tweet_results"] + tweet = tweet["items"][0]["item"]["itemContent"]["tweet_results"] + + elif "itemContent" in tweet.keys(): + if "tweet_results" in tweet["itemContent"]: + tweet = tweet["itemContent"]["tweet_results"] + else: + save_errored_tweet( + tweet, "Error getting [itemContent][tweet_results] key in parse_tweet()" + ) + return + # For long tweets + if "note_tweet" in tweet: + is_long_tweet = True + tweet_text = tweet["note_tweet"]["note_tweet_results"]["result"]["text"] + tweet_entities = tweet["note_tweet"]["note_tweet_results"]["result"][ + "entity_set" + ] + media = tweet["note_tweet"]["note_tweet_results"]["result"]["media"] + tweet_id = tweet["rest_id"] + + try: + tweet = tweet["result"] + except KeyError: + save_errored_tweet(tweet, "Error getting result key in parse_tweet()") + return + + # Ignore Tweets that are older than the latest tweet + if not is_long_tweet: + if "legacy" not in tweet: + if "tweet" not in tweet: + save_errored_tweet(tweet, "Error getting tweet key in parse_tweet()") + return + + tweet_id = int(tweet["tweet"]["rest_id"]) + else: + tweet_id = int(tweet["legacy"]["id_str"]) + + if "core" not in tweet: + if "tweet" in tweet: + tweet = tweet["tweet"] + else: + save_errored_tweet( + tweet, "Error getting [core][tweet] key in parse_tweet()" + ) + return + + # So we can use this function recursively + if update_tweet_id: + # Skip this tweet + if tweet_id <= util.vars.latest_tweet_id: + return + util.vars.latest_tweet_id = tweet_id + + # Get user info + user_name = get_user_info(tweet, "name") # The name of the account (not @username) + user_screen_name = get_user_info(tweet, "screen_name") # The @username + user_img = get_user_info(tweet, "profile_image_url_https") + + # Media + media = [] + media_types = [] + + if not is_long_tweet: + if "legacy" in tweet.keys(): + if "extended_entities" in tweet["legacy"].keys(): + if "media" in tweet["legacy"]["extended_entities"].keys(): + media = [ + image["media_url_https"] + for image in tweet["legacy"]["extended_entities"]["media"] + ] + # photo, video + media_types = [ + image["type"] + for image in tweet["legacy"]["extended_entities"]["media"] + ] + + # Remove t.co url from text + text = remove_twitter_url_at_end(tweet["legacy"]["full_text"]) + + # Tickers + tickers = get_entities(tweet, "symbols") + + # Hashtags + hashtags = get_entities(tweet, "hashtags") + + else: + text = tweet_text + + # Is this correct? + if "inline_media" in media.keys(): + media = [image["media_url_https"] for image in media["inline_media"]] + media_types = [image["type"] for image in media["inline_media"]] + tickers = tweet_entities["symbols"] + hashtags = tweet_entities["hashtags"] + + # Tweet url + tweet_url = f"https://twitter.com/user/status/{tweet_id}" + + quoted_status_result = tweet.get("quoted_status_result", False) + retweeted_status_result = tweet.get("legacy", {}).get( + "retweeted_status_result", False + ) + + e_title = f"{user_name} tweeted" + + if quoted_status_result or retweeted_status_result or reply: + result = quoted_status_result or retweeted_status_result or reply + replied_tweet = parse_tweet(result) + + # If parse_tweet errors it returns None + if replied_tweet: + ( + r_text, + r_user_name, + r_user_screen_name, + _, + _, + r_media, + r_tickers, + r_hashtags, + _, + r_media_types, + ) = replied_tweet + + if reply and replied_tweet: + e_title = f"{util.vars.custom_emojis['reply']} {user_name} replied to {r_user_name}" + text = "\n".join(map(lambda line: "> " + line, text.split("\n"))) + text = f"> [@{r_user_screen_name}](https://twitter.com/{r_user_screen_name}):\n{text}\n\n{r_text}" + + # Add text on top + if quoted_status_result and replied_tweet: + e_title = f"{util.vars.custom_emojis['quote_tweet']} {user_name} quote tweeted {r_user_name}" + q_text = "\n".join(map(lambda line: "> " + line, r_text.split("\n"))) + text = f"{text}\n\n> [@{r_user_screen_name}](https://twitter.com/{r_user_screen_name}):\n{q_text}" + + if retweeted_status_result and replied_tweet: + e_title = f"{util.vars.custom_emojis['retweet']} {user_name} retweeted {r_user_name}" + # Use the full retweeted text (otherwise the tweet text is cut off) + text = r_text + + media += r_media + media_types += r_media_types + tickers += r_tickers + hashtags += r_hashtags + + # Replace &amp; etc. + text = text.replace("&amp;", "&").replace("&gt;", ">").replace("&lt;", "<") + + # Convert media, tickers, hasthtags to sets to remove duplicates + media = list(set(media)) + tickers = list(set(tickers)) + hashtags = list(set(hashtags)) + + # tickers and hashtags all uppercase + tickers = [ticker.upper() for ticker in tickers] + hashtags = [hashtag.upper() for hashtag in hashtags if hashtag != "CRYPTO"] + + # Create the embed title + + return ( + text, + user_name, + user_screen_name, + user_img, + tweet_url, + media, + tickers, + hashtags, + e_title, + media_types, + )
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/sentiment_analyis.html b/_modules/util/sentiment_analyis.html new file mode 100644 index 00000000..702d4674 --- /dev/null +++ b/_modules/util/sentiment_analyis.html @@ -0,0 +1,524 @@ + + + + + + + + + + util.sentiment_analyis — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.sentiment_analyis

+##> Imports
+# > Standard libaries
+from __future__ import annotations
+
+import re
+
+# > Third party libraries
+import discord
+from transformers import AutoTokenizer, BertForSequenceClassification, pipeline
+
+# Load model
+model = BertForSequenceClassification.from_pretrained(
+    "StephanAkkerman/FinTwitBERT-sentiment",
+    num_labels=3,
+    id2label={0: "NEUTRAL", 1: "BULLISH", 2: "BEARISH"},
+    label2id={"NEUTRAL": 0, "BULLISH": 1, "BEARISH": 2},
+    cache_dir="models/",
+)
+model.config.problem_type = "single_label_classification"
+tokenizer = AutoTokenizer.from_pretrained(
+    "StephanAkkerman/FinTwitBERT-sentiment",
+    cache_dir="models/",
+    add_special_tokens=True,
+)
+model.eval()
+pipe = pipeline("text-classification", model=model, tokenizer=tokenizer)
+
+label_to_emoji = {
+    "NEUTRAL": "🦆",
+    "BULLISH": "🐂",
+    "BEARISH": "🐻",
+}
+
+color_table = {
+    "🦆": discord.Colour.lighter_grey(),
+    "🐂": discord.Colour.green(),
+    "🐻": discord.Colour.red(),
+}
+
+
+
+[docs] +def preprocess_text(tweet: str) -> str: + # Replace URLs with URL token + tweet = re.sub(r"http\S+", "[URL]", tweet) + + # Replace @mentions with @USER token + tweet = re.sub(r"@\S+", "@USER", tweet) + + return tweet
+ + + +
+[docs] +def classify_sentiment(text: str) -> str: + """ + Uses the text of a tweet to classify the sentiment of the tweet. + + Parameters + ---------- + text : str + The text of the tweet. + + Returns + ------- + np.ndarray + The probability of the tweet being bullish, neutral, or bearish. + """ + + label = pipe(preprocess_text(text))[0].get("label") + emoji = label_to_emoji[label] + + return emoji
+ + + +
+[docs] +def add_sentiment(e: discord.Embed, text: str) -> tuple[discord.Embed, str]: + """ + Adds sentiment to a discord embed, based on the given text. + + Parameters + ---------- + e : discord.Embed + The embed to add the sentiment to. + text : str + The text to classify the sentiment of. + + Returns + ------- + tuple[discord.Embed, str] + discord.Embed + The embed with the sentiment added. + str + The sentiment of the tweet. + """ + + # Remove quote tweet formatting + emoji = classify_sentiment(text.split("\n\n> [@")[0]) + + # Change color based on sentiment + e.colour = color_table[emoji] + + return e, emoji
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/ticker_classifier.html b/_modules/util/ticker_classifier.html new file mode 100644 index 00000000..39404f6a --- /dev/null +++ b/_modules/util/ticker_classifier.html @@ -0,0 +1,643 @@ + + + + + + + + + + util.ticker_classifier — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.ticker_classifier

+##> Imports
+# > Standard libaries
+from __future__ import annotations
+
+from typing import List, Optional
+
+from util.cg_data import get_coin_info
+
+# Local dependencies
+from util.tv_data import tv
+from util.tv_symbols import currencies
+from util.yf_data import get_stock_info
+
+
+
+[docs] +async def get_financials(ticker: str, website: str): + if "coingecko" in website: + _, _, _, price, change, _ = await get_coin_info(ticker) + four_h_ta, one_d_ta = tv.get_tv_TA(ticker, "crypto") + elif "yahoo" in website: + _, _, _, price, change, _ = await get_stock_info(ticker) + four_h_ta, one_d_ta = tv.get_tv_TA(ticker, "stock") + elif "forex" in website: + _, _, _, price, change, _ = await get_stock_info(ticker, "forex") + four_h_ta, one_d_ta = tv.get_tv_TA(ticker, "forex") + + return price, change, four_h_ta, one_d_ta
+ + + +
+[docs] +async def get_best_guess(ticker: str, asset_type: str): + """ + Gets the best guess of the ticker. + + Parameters + ---------- + ticker : str + The ticker mentioned in a tweet, e.g. BTC + asset_type : str + The guessed asset type, this can be crypto, stock or forex. + + Returns + ------- + tuple + The data of the best guess + """ + + get_TA = False + four_h_ta = one_d_ta = None + + if asset_type == "crypto" and ticker.endswith("BTC") and ticker != "BTC": + get_TA = True + ticker = ticker[:-3] + + if asset_type == "crypto": + ( + volume, + website, + exchange, + price, + change, + base_sym, + ) = await get_coin_info(ticker) + + elif asset_type == "stock": + ( + volume, + website, + exchange, + price, + change, + base_sym, + ) = await get_stock_info(ticker) + + elif asset_type == "forex": + if ticker in currencies: + return ( + 100000, + "https://www.tradingview.com/ideas/eur/?forex", + "forex", + None, + None, + None, + None, + ticker, + True, + ) + ( + volume, + website, + exchange, + price, + change, + base_sym, + ) = await get_stock_info(ticker, asset_type) + + if price > 0: + four_h_ta, one_d_ta = tv.get_tv_TA(ticker, "forex") + return ( + volume, + website, + exchange, + price, + change, + four_h_ta, + one_d_ta, + base_sym, + True, + ) + + # If volume of the crypto is bigger than 1,000,000, it is likely a crypto + # Stupid Tessla Coin https://www.coingecko.com/en/coins/tessla-coin + if volume > 1000000: + get_TA = True + + # Set the TA data, only if volume is high enough + if get_TA: + if base_sym is None: + print("No base symbol found for", ticker) + base_sym = ticker + four_h_ta, one_d_ta = tv.get_tv_TA(base_sym, asset_type) + + return ( + volume, + website, + exchange, + price, + change, + four_h_ta, + one_d_ta, + base_sym, + get_TA, + )
+ + + +
+[docs] +async def classify_ticker( + ticker: str, majority: str +) -> Optional[tuple[float, str, List[str], float, str, str, str]]: + """ + Main function to classify the ticker as crypto or stock. + + Parameters + ---------- + ticker : str + The ticker of the coin or stock. + majority : str + The guessed majority of the ticker. + + Returns + ------- + Optional[tuple[float, str, List[str], float, str, str]] + float + The volume of the coin or stock. + str + The website of the coin or stock. + list[str] + The exchanges of the coin or stock. + float + The price of the coin or stock. + str + The 24h price change of the coin or stock. + str + The four hour technical analysis using TradingView. + str + The daily technical analysis using TradingView. + str + The base ticker. + """ + + # Try forex first + forex_data = await get_best_guess(ticker, "forex") + if forex_data[-1]: + return forex_data[:-1] + + # If the majority is crypto or unkown check if the ticker is a crypto + if majority == "crypto": + crypto_data = await get_best_guess(ticker, "crypto") + + if crypto_data[-1]: + return crypto_data[:-1] + + stock_data = await get_best_guess(ticker, "stock") + + elif majority == "stocks": + stock_data = await get_best_guess(ticker, "stock") + + if stock_data[-1]: + return stock_data[:-1] + + crypto_data = await get_best_guess(ticker, "crypto") + else: + crypto_data = await get_best_guess(ticker, "crypto") + stock_data = await get_best_guess(ticker, "stock") + + # If it was not the majority, compare the data + c_volume = crypto_data[0] + s_volume = stock_data[0] + + if c_volume > s_volume and c_volume > 50000: + if crypto_data[5] is None: + four_h_ta, one_d_ta = tv.get_tv_TA(ticker, "crypto") + + crypto_data = list(crypto_data) + crypto_data[5] = four_h_ta + crypto_data[6] = one_d_ta + tuple(crypto_data) + + return crypto_data[:-1] + + elif c_volume < s_volume: + if stock_data[5] is None: + four_h_ta, one_d_ta = tv.get_tv_TA(ticker, "stock") + + stock_data = list(stock_data) + stock_data[5] = four_h_ta + stock_data[6] = one_d_ta + tuple(stock_data) + + return stock_data[:-1]
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/trades_msg.html b/_modules/util/trades_msg.html new file mode 100644 index 00000000..fd0cc018 --- /dev/null +++ b/_modules/util/trades_msg.html @@ -0,0 +1,626 @@ + + + + + + + + + + util.trades_msg — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.trades_msg

+##> Imports
+import datetime
+
+import ccxt.pro
+
+# > Discord dependencies
+import discord
+import pandas as pd
+
+import util.trades_msg
+
+# Local dependencies
+import util.vars
+from util.db import get_db, update_db
+from util.exchange_data import get_buying_price, get_data, get_usd_price
+from util.formatting import format_change
+from util.vars import stables
+
+
+
+[docs] +async def on_msg( + msg: list, + exchange: ccxt.pro.Exchange, + trades_channel: discord.TextChannel, + row: pd.Series, + user: discord.User, +) -> None: + """ + This function is used to handle the incoming messages from the binance websocket. + + Parameters + ---------- + msg : str + The message that is received from the binance websocket. + + Returns + ------- + None + """ + + msg = msg[0] + sym = msg["symbol"] # BNB/USDT + orderType = msg["type"] # market, limit, stop, stop limit + side = msg["side"] # buy, sell + price = float(round(msg["price"], 4)) + amount = float(round(msg["amount"], 4)) + cost = float(round(msg["cost"], 4)) # If /USDT, then this is the USD value + + # Get the value in USD + usd = price + base = sym.split("/")[0] + quote = sym.split("/")[1] + if quote not in stables: + usd, change = await get_usd_price(exchange, base) + + # Get profit / loss if it is a sell + buying_price = None + if side == "sell": + buying_price = await get_buying_price(exchange, sym, True) + + # Send it in the discord channel + await util.trades_msg.trades_msg( + exchange.id, + trades_channel, + user, + sym, + side, + orderType, + price, + amount, + round(usd * amount, 2), + buying_price, + ) + + # Assets db: asset, owned (quantity), exchange, id, user + assets_db = get_db("assets") + + # Drop all rows for this user and exchange + updated_assets_db = assets_db.drop( + assets_db[ + (assets_db["id"] == row["id"]) & (assets_db["exchange"] == exchange.id) + ].index + ) + + assets_db = pd.concat([updated_assets_db, await get_data(row)]).reset_index( + drop=True + ) + + update_db(assets_db, "assets") + util.vars.assets_db = assets_db
+ + # Maybe post the updated assets of this user as well + + +
+[docs] +async def trades_msg( + exchange: str, + channel: discord.TextChannel, + user: discord.User, + symbol: str, + side: str, + orderType: str, + price: float, + quantity: float, + usd: float, + buying_price: float = None, +) -> None: + """ + Formats the Discord embed that will be send to the dedicated trades channel. + + Parameters + ---------- + exchange : str + The name of the exchange, currently only supports "binance", "kucoin" and "stocks". + channel : discord.TextChannel + The channel that the message will be sent to. + user : discord.User + The user that the message will be sent from. + symbol : str + The symbol that has been traded. + side : str + The side of the trade, either "BUY" or "SELL". + orderType : str + The type of order, for instance "LIMIT" or "MARKET". + price : float + The price of the trade. + quantity : float + The amount traded. + usd : float + The worth of the trade in US dollar. + + Returns + ------- + None + """ + + # Same as in formatting.py + if exchange == "binance": + color = 0xF0B90B + icon_url = ( + "https://upload.wikimedia.org/wikipedia/commons/5/57/Binance_Logo.png" + ) + url = f"https://www.binance.com/en/trade/{symbol}" + elif exchange == "kucoin": + color = 0x24AE8F + icon_url = "https://yourcryptolibrary.com/wp-content/uploads/2021/12/Kucoin-exchange-logo-1.png" + url = f"https://www.kucoin.com/trade/{symbol}" + else: + color = 0x720E9E + icon_url = ( + "https://s.yimg.com/cv/apiv2/myc/finance/Finance_icon_0919_250x252.png" + ) + url = f"https://finance.yahoo.com/quote/{symbol}" + + e = discord.Embed( + title=f"{orderType.capitalize()} {side.lower()} {quantity} {symbol}", + description="", + color=color, + url=url, + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + # Set the embed fields + e.set_author(name=user.name, icon_url=user.display_avatar.url) + + # If the quote is USD, then the price is the USD value + e.add_field( + name="Price", + value=f"${price}" if symbol.endswith(tuple(stables)) else price, + inline=True, + ) + + if buying_price and buying_price != 0: + price_change = price - buying_price + + if price_change != 0: + percent_change = round((price_change / buying_price) * 100, 2) + else: + percent_change = 0 + + percent_change = format_change(percent_change) + profit_loss = f"${round(price_change * quantity, 2)} ({percent_change})" + + e.add_field( + name="Profit / Loss", + value=profit_loss, + inline=True, + ) + else: + e.add_field(name="Amount", value=quantity, inline=True) + + # If we know the USD value, then add it + if usd != 0: + e.add_field( + name="$ Worth", + value=f"${usd}", + inline=True, + ) + + e.set_footer(text="\u200b", icon_url=icon_url) + + await channel.send(embed=e) + + # Tag the person + if orderType.upper() != "MARKET": + await channel.send(f"<@{user.id}>")
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/tv_data.html b/_modules/util/tv_data.html new file mode 100644 index 00000000..3ff00ab4 --- /dev/null +++ b/_modules/util/tv_data.html @@ -0,0 +1,822 @@ + + + + + + + + + + util.tv_data — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.tv_data

+## > Imports
+# > Standard libaries
+from __future__ import annotations
+
+import json
+import random
+import re
+import string
+import traceback
+from typing import List, Optional
+
+# > 3rd party dependencies
+import aiohttp
+import pandas as pd
+from tradingview_ta import Interval, TA_Handler
+
+# > Local dependencies
+import util.vars
+from util.tv_symbols import all_forex_indices, crypto_indices, stock_indices
+from util.vars import get_json_data
+
+
+
+[docs] +async def get_tv_ticker_data(url, append_to=None): + data = await get_json_data(url) + + if not data or data == {} or "data" not in data.keys(): + return pd.DataFrame() + + # Convert data to pandas df + tv_data = pd.DataFrame(data["data"]).drop(columns=["d"]) + + if append_to: + # This adds additional information to the dataframe + tv_data = pd.concat([tv_data, pd.DataFrame(append_to, columns=["s"])]) + + # Split the information in exchange and stock + tv_data[["exchange", "stock"]] = tv_data["s"].str.split(":", n=1, expand=True) + + return tv_data
+ + + +
+[docs] +class TV_data: + """ + This class is used to get the current price, 24h change, and volume of a stock. + It also includes methods to get the TradingView TA data. + """ + + def __init__(self) -> None: + self.stock_indices_without_exch = [sym.split(":")[1] for sym in stock_indices] + self.crypto_indices_without_exch = [sym.split(":")[1] for sym in crypto_indices] + self.forex_indices_without_exch = [ + sym.split(":")[1] for sym in all_forex_indices + ] + +
+[docs] + async def on_msg( + self, ws: aiohttp.ClientWebSocketResponse, msg + ) -> Optional[tuple[float, float, float]]: + """ + Parses the message from the TradingView API. + + Returns + ------- + Optional[tuple[float, float, float, str]] + float + The current price. + float + The current 24h change. + float + The current volume. + """ + + try: + # Try again + if '"m":' not in msg: + return None + elif Res := re.findall("^.*?({.*)$", msg): + jsonRes = json.loads(Res[0].split("~m~")[0]) + if "m" in jsonRes.keys(): + if jsonRes["m"] == "qsd": + try: + price = float(jsonRes["p"][1]["v"]["lp"]) + change = float(jsonRes["p"][1]["v"]["ch"]) + volume = float(jsonRes["p"][1]["v"]["volume"]) + except KeyError: + print("KeyError in TradingView ws_data") + return None + + if price != 0: + perc_change = round((change / price) * 100, 2) + else: + print("TradingView returns price=0") + return None + + return price, perc_change, volume + else: + # ping packet + pingStr = re.findall(".......(.*)", msg) + if len(pingStr) != 0: + pingStr = pingStr[0] + ws.send_str("~m~" + str(len(pingStr)) + "~m~" + pingStr) + + return None + except Exception: + print(traceback.format_exc())
+ + +
+[docs] + async def sendMessage( + self, ws: aiohttp.ClientWebSocketResponse, func: str, args: List[str] + ) -> None: + """ + Sends a message to the TradingView API. + This needs to be done before any data can be retrieved. + + Parameters + ---------- + ws : aiohttp.ClientWebSocketResponse + The websocket object to send the message from. + func : str + The function to call, all start with ``quote_`` followed by the function name. + args : List[str] + The list of arguments to send in the message. + """ + + as_json = json.dumps({"m": func, "p": args}, separators=(",", ":")) + prepended = "~m~" + str(len(as_json)) + "~m~" + as_json + await ws.send_str(prepended)
+ + +
+[docs] + def get_usd_info(self, tv_crypto, symbol: str, suffix: str): + if not symbol.endswith(suffix): + # If it crypto try adding USD or USDT + crypto_USD = tv_crypto.loc[tv_crypto["stock"] == symbol + suffix] + + if not crypto_USD.empty: + return ( + crypto_USD["exchange"].values[0], + "crypto", + crypto_USD["stock"].values[0], + )
+ + +
+[docs] + def get_symbol_data( + self, symbol: str, asset: str + ) -> Optional[tuple[str, str, str]]: + """ + Helper function to get the symbol data from the TradingView API. + This data included the exchange and market this symbol is traded on. + + Parameters + ---------- + symbol : str + The ticker of the stock / crypto. + asset : str + The type of asset, either "stock" or "crypto". + + Returns + ------- + Optional[tuple[str, str, str]] + str + The exchange the symbol is traded on, e.g. "FTX" or "Binance". + str + The market the symbol is traded on, e.g. "crypto", "america", "forex". + str + The symbol itself. + """ + + tv_stocks = util.vars.stocks + tv_crypto = util.vars.crypto + tv_forex = util.vars.forex + + if asset == "stock": + stock = tv_stocks.loc[tv_stocks["stock"] == symbol] + if not stock.empty: + return stock["exchange"].values[0], "america", symbol + + elif asset == "forex": + forex = tv_forex.loc[tv_forex["stock"] == symbol] + if not forex.empty: + return forex["exchange"].values[0], "forex", symbol + + elif asset == "crypto": + crypto = tv_crypto.loc[tv_crypto["stock"] == symbol] + if not crypto.empty: + return crypto["exchange"].values[0], "crypto", symbol + else: + # Iterate over some USD suffixes + for s in ["USD", "USDT", "USDTPERP"]: + if data := self.get_usd_info(tv_crypto, symbol, s): + return data
+ + +
+[docs] + async def get_tv_data( + self, symbol: str, asset: str + ) -> Optional[tuple[float, float, float, str, str]]: + """ + Gets the current price, volume, 24h change, and TA data from the TradingView API. + + Parameters + ---------- + symbol: string + The ticker of the stock / crypto, e.g. "AAPL" or "BTCUSDT". + asset: string + The type of asset, either "stock", "crypto", or "forex". + + Returns + ------- + Optional[tuple[float, float, float, str]] + float + The current price. + float + The current 24h change. + float + The current volume. + str + The exchange that this symbol is listed on. + str + The url to the TradingView chart for this symbol. + """ + + if asset == "stock": + website_suffix = "/?yahoo" + elif asset == "forex": + website_suffix = "/?forex" + elif asset == "crypto": + website_suffix = "/?coingecko" + + website = f"https://www.tradingview.com/symbols/{symbol}{website_suffix}" + + try: + symbol_data = self.get_symbol_data(symbol, asset) + if symbol_data is not None: + # Format it "exchange:symbol" + exchange = symbol_data[0] + symbol = f"{exchange}:{symbol_data[2]}" + website = f"https://www.tradingview.com/symbols/{symbol_data[2]}{website_suffix}" + + else: + return (0, None, 0, None, website) + + # Create a session + session = aiohttp.ClientSession() + + async with session.ws_connect( + url="wss://data.tradingview.com/socket.io/websocket", + headers={"Origin": "https://data.tradingview.com"}, + ) as ws: + # This is mandatory to get the data + auth_str = "qs_" + "".join( + random.choice(string.ascii_lowercase) for i in range(12) + ) + + # Send messages via websocket + await self.sendMessage(ws, "quote_create_session", [auth_str]) + await self.sendMessage( + ws, "quote_set_fields", [auth_str, "ch", "lp", "volume"] + ) + await self.sendMessage(ws, "quote_add_symbols", [auth_str, symbol]) + + counter = 0 + + # Check for response + async for msg in ws: + counter += 1 + + if msg.type == aiohttp.WSMsgType.TEXT: + resp = await self.on_msg(ws, msg.data) + + if resp is not None: + await ws.close() + await session.close() + # Convert to USD volume if asset is crypto + return ( + float(resp[0]), + float(resp[1]), + resp[0] * resp[2] if asset == "crypto" else resp[2], + exchange.lower(), + website, + ) + + elif counter == 3: + await session.close() + return (0, None, 0, None, website) + + elif msg.type == aiohttp.WSMsgType.ERROR: + # self.restart_sockets() + print("TradingView websocket Error") + await session.close() + return (0, None, 0, None, website) + + except aiohttp.ClientConnectionError: + print("Temporary TradingView websocket error") + + except Exception: + print(traceback.format_exc()) + + return (0, None, 0, None, website)
+ + +
+[docs] + def format_analysis(self, analysis: dict) -> str: + """ + Simple helper function to format the TA data into one string. + + Parameters + ---------- + analysis : dict + The original TA data from the TradingView API. + + Returns + ------- + str + The formatted TA data. + """ + + return f"{analysis['RECOMMENDATION']}\n{analysis['BUY']}📈 {analysis['NEUTRAL']}⌛️ {analysis['SELL']}📉"
+ + +
+[docs] + def get_tv_TA(self, symbol: str, asset: str) -> Optional[tuple[str, str]]: + """ + Gets the current TA (technical analysis) data from the TradingView API. + + Parameters + ---------- + symbol : str + The ticker of the stock / crypto. + asset : str + The type of asset, either "stock", "crypto" or "forex". + + Returns + ------- + Optional[tuple[str,str]] + The 4h and 1d TA data as formatted strings. + """ + + # There is no TA for indices + if ( + symbol in self.stock_indices_without_exch + or symbol in self.crypto_indices_without_exch + or symbol in self.forex_indices_without_exch + ): + return None, None + + symbol_data = self.get_symbol_data(symbol, asset) + four_h_analysis = one_d_analysis = None + + # Get the TradingView TA for symbol + # Interval can be 1m, 5m, 15m, 30m, 1h, 2h, 4h, 1d, 1W, 1M + if symbol_data is not None: + exchange, market, symbol = symbol_data + + # Wait max 5 sec + try: + four_h_analysis = TA_Handler( + symbol=symbol, + screener=market, + exchange=exchange, + interval=Interval.INTERVAL_4_HOURS, + timeout=5, + ).get_analysis() + + one_d_analysis = TA_Handler( + symbol=symbol, + screener=market, + exchange=exchange, + interval=Interval.INTERVAL_1_DAY, + timeout=5, + ).get_analysis() + + except Exception as e: + print(f"TradingView TA error for ticker: {symbol}, error:", e) + return None, None + + if four_h_analysis: + four_h_analysis = self.format_analysis(four_h_analysis.summary) + + if one_d_analysis: + one_d_analysis = self.format_analysis(one_d_analysis.summary) + + # Format the analysis + return four_h_analysis, one_d_analysis + + return None, None
+
+ + + +tv = TV_data() +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/tweet_embed.html b/_modules/util/tweet_embed.html new file mode 100644 index 00000000..32c0f53d --- /dev/null +++ b/_modules/util/tweet_embed.html @@ -0,0 +1,827 @@ + + + + + + + + + + util.tweet_embed — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.tweet_embed

+## > Imports
+# > Standard libaries
+from __future__ import annotations
+
+import datetime
+from typing import List
+
+# Discord imports
+import discord
+import numpy as np
+
+# 3rd party imports
+import pandas as pd
+from discord.ext import commands
+
+# Local dependencies
+import util.vars
+from cogs.loops.overview import Overview
+from util.db import merge_and_update, remove_old_rows, update_tweet_db
+from util.sentiment_analyis import add_sentiment
+from util.ticker_classifier import classify_ticker, get_financials
+from util.vars import data_sources, filter_dict
+
+tweet_overview = None
+
+
+
+[docs] +async def make_tweet_embed( + text: str, + user_name: str, + profile_pic: str, + url: str, + images: List[str], + tickers: List[str], + hashtags: List[str], + e_title: str, + media_types: List[str], + bot: commands.Bot, +) -> tuple[discord.Embed, str, str, list, list]: + """ + Pre-processing the tweet data before uploading it to the Discord channels. + This function creates the embed object and tags the user after it is correctly uploaded. + + Parameters + ---------- + text : str + The text of the tweet. + user : str + The user that posted this tweet. + profile_pic : str + The url to the profile pic of the user. + url : str + The url to the tweet. + images : list + The images contained in this tweet. + tickers : list + The tickers contained in this tweet (i.e. $BTC). + hashtags : list + The hashtags contained in this tweet. + retweeted_user : str + The user that was retweeted by this tweet. + bot : commands.Bot + Discord bot object. + + Returns + ------- + None + """ + + category = None + base_symbols = [] + + # Ensure the tickers are unique + symbols = get_clean_symbols(tickers, hashtags)[:24] + + e = make_embed(symbols, url, text, profile_pic, images, e_title, media_types) + + # Max 25 fields + if symbols: + print("Adding financials to tweet embed...") + e, category, base_symbols = await add_financials( + e, symbols, tickers, text, user_name, bot + ) + + return e, category, base_symbols
+ + + +
+[docs] +def make_embed( + symbols, url, text, profile_pic, images, e_title, media_types: List[str] +) -> discord.Embed: + # Set the properties of the embed + e = discord.Embed( + title=embed_title(e_title, symbols), + url=url, + description=text, + color=data_sources["twitter"]["color"], + timestamp=datetime.datetime.now(datetime.timezone.utc), + ) + + e.set_thumbnail(url=profile_pic) + + # Set image if an image is included in the tweet + if images: + e.set_image(url=images[0]) + + footer_text = "\u200b" + + if "video" in media_types: + footer_text = "Video" + elif "animated_gif" in media_types: + footer_text = "GIF" + + # Set the twitter icon as footer image + e.set_footer( + text=footer_text, + icon_url=data_sources["twitter"]["icon"], + ) + + return e
+ + + +
+[docs] +def embed_title(e_title: str, tickers: list) -> str: + if not tickers: + return e_title + + title = f"{e_title} about {', '.join(tickers)}" + + # The max length of the title is 256 characters + if len(title) > 256: + title = title[:253] + "..." + + return title
+ + + +
+[docs] +async def add_financials( + e: discord.Embed, + symbols: List[str], + tickers: List[str], + text: str, + user: str, + bot: commands.Bot, +) -> tuple[discord.Embed, str, List[str]]: + """ + Adds the financial data to the embed and returns the corresponding category. + + Parameters + ---------- + e : discord.Embed + The embed to add the data to. + symbols : List[str] + The symbols (tickers + hashtags) in the tweet. + tickers : List[str] + The tickers in the tweet. + text : str + The text of the tweet. + user : str + The user that tweeted. + bot : commands.Bot + The bot object, used for getting the custom emojis. + + Returns + ------- + tuple[discord.Embed, str, str] + discord.Embed + The embed with the data added. + str + The category of the tweet. + List[str] + The base symbols of the tickers. + """ + global tweet_overview + + # In case multiple tickers get send + crypto = stocks = forex = 0 + + base_symbols = [] + categories = [] + do_last = [] + classified_tickers = [] + changes = [] + + if not util.vars.classified_tickers.empty: + # Drop tickers older than 3 days + util.vars.classified_tickers = remove_old_rows(util.vars.classified_tickers, 3) + classified_tickers = util.vars.classified_tickers["ticker"].tolist() + + for ticker in symbols: + if crypto > stocks and crypto > forex: + majority = "crypto" + elif stocks > crypto and stocks > forex: + majority = "stocks" + elif forex > crypto and forex > stocks: + majority = "forex" + else: + majority = "Unknown" + + # Get the information about the ticker + if ticker not in classified_tickers: + ticker_info = await classify_ticker(ticker, majority) + if ticker_info: + ( + _, + website, + exchanges, + price, + change, + four_h_ta, + one_d_ta, + base_symbol, + ) = ticker_info + + # Skip if this ticker has been done before, for instance in tweets containing Solana and SOL + if base_symbol in base_symbols: + continue + + if exchanges is None: + exchanges = [] + print("No exchanges found for", ticker) + + # Convert info to a dataframe + df = pd.DataFrame( + [ + { + "ticker": ticker, + "website": website, + # Db cannot handle lists, so we convert them to strings + "exchanges": ( + ";".join(exchanges) if len(exchanges) > 0 else "" + ), + "base_symbol": base_symbol, + "timestamp": datetime.datetime.now(), + } + ] + ) + + # Save the ticker info in a database + util.vars.classified_tickers = merge_and_update( + util.vars.classified_tickers, df, "classified_tickers" + ) + + else: + if ticker in tickers: + e.add_field(name=f"${ticker}", value=majority) + print( + f"No crypto or stock match found for ${ticker} in {user}'s tweet at {datetime.datetime.now().strftime('%Y-%m-%d %H:%M')}" + ) + + # Go to next in symbols + continue + else: + ticker_info = util.vars.classified_tickers[ + util.vars.classified_tickers["ticker"] == ticker + ] + website = ticker_info["website"].values[0] + exchanges = ticker_info["exchanges"].values[0] + # Convert string to list + exchanges = exchanges.split(";") + base_symbol = ticker_info["base_symbol"].values[0] + + # Still need the price, change, TA info + price, change, four_h_ta, one_d_ta = await get_financials(ticker, website) + + title = f"${ticker}" + + # Add to base symbol list to prevent duplicates + base_symbols.append(base_symbol) + + if isinstance(change, list) and len(change) == 1: + changes.append(change[-1]) + else: + changes.append(change) + + # Determine if this is a crypto or stock + if website: + if "coingecko" in website: + crypto += 1 + categories.append("crypto") + for x in exchanges: + if x in util.vars.custom_emojis.keys(): + title = f"{title} {util.vars.custom_emojis[x]}" + + if "yahoo" in website: + stocks += 1 + categories.append("stocks") + if "forex" in website: + forex += 1 + categories.append("forex") + else: + # Default category is crypto + categories.append("crypto") + + # If there is no TA for a symbol, add it at the end of the embed + if four_h_ta is None: + do_last.append((title, change, price, website)) + continue + + # Add the field with hyperlink + e.add_field( + name=title, value=get_description(change, price, website), inline=True + ) + + e.add_field(name="4h TA", value=four_h_ta, inline=True) + + if one_d_ta: + e.add_field(name="1d TA", value=one_d_ta, inline=True) + + for title, change, price, website in do_last: + e.add_field( + name=title, value=get_description(change, price, website), inline=True + ) + + # Finally add the sentiment to the embed + if base_symbols: # or if categories: + e, prediction = add_sentiment(e, text) + else: + prediction = None + + # Decide the category of this tweet + if crypto == 0 and stocks == 0 and forex == 0: + category = None + else: + category = ("crypto", "stocks", "forex")[np.argmax([crypto, stocks, forex])] + + # If there are base symbols, add them to the database + # Also post the overview of mentioned tickers + if base_symbols: + update_tweet_db(base_symbols, user, prediction, categories, changes) + + if not tweet_overview: + tweet_overview = Overview(bot) + + await tweet_overview.overview(category, base_symbols, prediction) + + # Return just the prediction without emoji + return e, category, base_symbols
+ + + +
+[docs] +def get_clean_symbols(tickers, hashtags): + # Remove #NFT from the list + hashtags = [hashtag for hashtag in hashtags if hashtag not in ["NFT", "CRYPTO"]] + + # First remove the duplicates + symbols = list(set(tickers + hashtags)) + + clean_symbols = [] + + # Check the filter dict + for symbol in symbols: + # Filter beforehand + if symbol in filter_dict.keys(): + # For instance Ethereum -> ETH + new_sym = filter_dict[symbol] + # However if ETH is in there we do not want to have it twice + if new_sym not in clean_symbols: + clean_symbols.append(new_sym) + else: + clean_symbols.append(symbol) + + return clean_symbols
+ + + +
+[docs] +def format_description( + AH: bool, change: list, price: list, website: str, i: int +) -> str: + if AH: + return f"[AH: ${price[i]}\n({change[i]})]({website})\n" + else: + return f"[${price[i]}\n({change[i]})]({website})"
+ + + +
+[docs] +def get_description(change, price, website): + if not change and not price: + return "\u200b" + + # Change can be a list (if the information is from Yahoo Finance) or a string + if type(change) == list and type(price) == list: + # If the length is 2 then we know the after-hour prices + if len(change) == 2 and len(price) == 2: + for i in range(len(change)): + if i == 0: + description = format_description(True, change, price, website, i) + else: + description += format_description(False, change, price, website, i) + else: + return format_description(False, change, price, website, 0) + + else: + return format_description(False, [change], [price], website, 0) + + return description
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/vars.html b/_modules/util/vars.html new file mode 100644 index 00000000..cdad7f8b --- /dev/null +++ b/_modules/util/vars.html @@ -0,0 +1,593 @@ + + + + + + + + + + util.vars — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.vars

+import json
+import os
+import sys
+
+import aiohttp
+import pandas as pd
+
+# > 3rd Party Dependencies
+import yaml
+
+# Read config.yaml content
+config_path = os.path.join(os.path.dirname(__file__), "..", "..", "config.yaml")
+with open(config_path, "r", encoding="utf-8") as f:
+    config = yaml.full_load(f)
+
+guild_name = (
+    os.getenv("DEBUG_GUILD")
+    if len(sys.argv) > 1 and sys.argv[1] == "-test"
+    else os.getenv("DISCORD_GUILD")
+)
+
+# Replace key by value
+filter_dict = {
+    "BITCOIN": "BTC",
+    "BTCD": "BTC.D",
+    "ETHEREUM": "ETH",
+    "ES_F": "ES=F",
+    "ES": "ES=F",
+    "NQ": "NQ=F",
+    "NQ_F": "NQ=F",
+    "CL_F": "CL=F",
+    "APPL": "AAPL",
+    "DEFI": "DEFIPERP",
+    "NVIDIA": "NVDA",
+}
+
+icon_url = (
+    "https://raw.githubusercontent.com/StephanAkkerman/fintwit-bot/main/img/icons/"
+)
+data_sources = {
+    "twitter": {"color": 0x1DA1F2, "icon": icon_url + "twitter.png"},
+    "yahoo": {"color": 0x720E9E, "icon": icon_url + "yahoo.png"},
+    "binance": {"color": 0xF0B90B, "icon": icon_url + "binance.png"},
+    "investing": {"color": 0xDC8F02, "icon": icon_url + "investing.png"},
+    "coingecko": {"color": 0x8AC14B, "icon": icon_url + "coingecko.png"},
+    "opensea": {"color": 0x3685DF, "icon": icon_url + "opensea.png"},
+    "coinmarketcap": {"color": 0x0D3EFD, "icon": icon_url + "cmc.ico"},
+    "playtoearn": {"color": 0x4792C9, "icon": icon_url + "playtoearn.png"},
+    "tradingview": {"color": 0x131722, "icon": icon_url + "tradingview.png"},
+    "coinglass": {"color": 0x000000, "icon": icon_url + "coinglass.png"},
+    "kucoin": {"color": 0x24AE8F, "icon": icon_url + "kucoin.png"},
+    "coinbase": {"color": 0x245CFC, "icon": icon_url + "coinbase.png"},
+    "unusualwhales": {"color": 0x000000, "icon": icon_url + "unusualwhales.png"},
+    "reddit": {"color": 0xFF3F18, "icon": icon_url + "reddit.png"},
+    "nasdaqtrader": {"color": 0x0996C7, "icon": icon_url + "nasdaqtrader.png"},
+    "stocktwits": {"color": 0xFFFFFF, "icon": icon_url + "stocktwits.png"},
+    "cryptocraft": {"color": 0x634C7B, "icon": icon_url + "cryptocraft.png"},
+    "barchart": {"color": 0x84C8C, "icon": icon_url + "barchart.png"},
+}
+
+# Stable coins
+# Could update this on startup:
+# https://www.binance.com/bapi/composite/v1/public/promo/cmc/cryptocurrency/category?id=604f2753ebccdd50cd175fc1&limit=10
+# Get info stored in ["data"]["body"]["data"]["coins"] to get this list
+stables = [
+    "USDT",
+    "USDC",
+    "BUSD",
+    "DAI",
+    "FRAX",
+    "TUSD",
+    "USDP",
+    "USDD",
+    "USDN",
+    "FEI",
+    "USD",
+    "USDTPERP",
+    "EUR",
+]
+
+# Init global database vars
+assets_db = None
+portfolio_db = None
+cg_db = None
+tweets_db = None
+options_db = None
+latest_tweet_id = 0
+
+# These variables save the TradingView tickers
+stocks = None
+crypto = None
+forex = None
+cfd = None
+
+nasdaq_tickers = None
+
+reddit_ids = pd.DataFrame()
+ideas_ids = pd.DataFrame()
+classified_tickers = pd.DataFrame()
+
+custom_emojis = {}
+
+
+
+[docs] +async def get_json_data( + url: str, + headers: dict = None, + cookies: dict = None, + json_data: dict = None, + text: bool = False, +) -> dict: + """ + Asynchronous function to get JSON data from a website. + + Parameters + ---------- + url : str + The URL to get the data from. + headers : dict, optional + The headers send with the get request, by default None. + + Returns + ------- + dict + The response as a dict. + """ + + try: + async with aiohttp.ClientSession(headers=headers, cookies=cookies) as session: + async with session.get(url, json=json_data) as r: + if text: + return await r.text() + else: + return await r.json() + except aiohttp.ClientError as e: + print(f"Error with get request for {url}.\nError: {e}") + except json.JSONDecodeError as e: + print(f"Error decoding JSON from {url}.\nError: {e}") + return {}
+ + + +
+[docs] +async def post_json_data( + url: str, + headers: dict = None, + data: dict = None, + json: dict = None, +) -> dict: + """ + Asynchronous function to post JSON data from a website. + + Parameters + ---------- + url : str + The URL to get the data from. + headers : dict, optional + The headers send with the post request, by default None. + + Returns + ------- + dict + The response as a dict. + """ + + try: + async with aiohttp.ClientSession(headers=headers) as session: + async with session.post(url, data=data, json=json) as r: + return await r.json(content_type=None) + except Exception as e: + print(f"Error with POST request for {url}.", "Error:", e) + + return {}
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_modules/util/yf_data.html b/_modules/util/yf_data.html new file mode 100644 index 00000000..c2c19567 --- /dev/null +++ b/_modules/util/yf_data.html @@ -0,0 +1,537 @@ + + + + + + + + + + util.yf_data — FinTwit Bot 1.2.0 documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + + + + + + + +
+
+
+
+
+ +
+ +
+ + + + + +
+
+ + + + + +
+ + + + + + + + + + + +
+ +
+ + +
+
+ +
+
+ +
+ +
+ + + + +
+ +
+ + +
+
+ + + + + +
+ +

Source code for util.yf_data

+##> Imports
+# > Standard libaries
+from __future__ import annotations
+
+from typing import List, Optional
+
+# > 3rd Party Dependencies
+from yahooquery import Ticker
+
+from util.afterhours import afterHours
+
+# Local dependencies
+from util.formatting import format_change
+from util.tv_data import tv
+
+
+
+[docs] +def yf_info(ticker: str, do_format_change: bool = True): + # try: + stock_info = Ticker(ticker, asynchronous=False).price + + # Test if the ticker is valid + if not isinstance(stock_info.get(ticker), dict): + return None + + stock_info = stock_info[ticker] + prices = [] + changes = [] + + # Helper function to format and append price data + def append_price_data(price_key, change_key): + price = stock_info.get(price_key) + change = stock_info.get(change_key, 0) + if do_format_change: + change = format_change(change) + if price and price != 0: + prices.append(price) + changes.append(change or "N/A") # Handle None or missing change + + # Determine which price to report based on market hours + if afterHours(): + append_price_data("preMarketPrice", "preMarketChangePercent") + append_price_data("regularMarketPrice", "regularMarketChangePercent") + + # Calculate volume + volume: float = ( + stock_info.get("regularMarketVolume", 0) * prices[-1] if prices else 0 + ) + + # Prepare return values + url: str = f"https://finance.yahoo.com/quote/{ticker}" + exchange: str = stock_info.get("exchange", []) + + return volume, url, exchange, prices, changes if changes else ["N/A"], ticker + + # TODO: ratelimit exception + # except Exception as e: + # print(f"Error in getting Yahoo Finance data for {ticker}: {e}") + + return None
+ + + +
+[docs] +async def get_stock_info( + ticker: str, asset_type: str = "stock", do_format_change: bool = True +) -> Optional[tuple[float, str, List[str], float, str, str]]: + """ + Gets the volume, website, exchanges, price, and change of the stock. + + Parameters + ---------- + ticker : str + The ticker of the stock. + asset_type : str + The type of asset, this can be stock or forex. + do_format_change : bool + Whether to format the change or not. + + Returns + ------- + Optional[tuple[float, str, List[str], float, str]] + float + The volume of the stock. + str + The website of the stock. + list[str] + The exchanges of the stock. + float + The price of the stock. + str + The 24h price change of the stock. + str + The ticker, to match the crypto function. + """ + + if asset_type == "stock": + stock_info = yf_info(ticker, do_format_change) + if stock_info and stock_info[0] > 0: # or price == [] + return stock_info + + # Check TradingView data + tv_data = await tv.get_tv_data(ticker, asset_type) + if tv_data: + # print(f"Could not find {ticker} on Yahoo Finance, using TradingView data.") + price, perc_change, volume, exchange, website = tv_data + + if do_format_change: + perc_change = format_change(perc_change) if perc_change else "N/A" + return ( + volume, + website, + exchange, + price, + perc_change, + ticker, + )
+ +
+ +
+ + + + + +
+ +
+
+
+ +
+ + + + +
+ + +
+
+
+ + + + + + + + \ No newline at end of file diff --git a/_sources/cogs.commands.rst.txt b/_sources/cogs.commands.rst.txt new file mode 100644 index 00000000..1eb4eb02 --- /dev/null +++ b/_sources/cogs.commands.rst.txt @@ -0,0 +1,61 @@ +cogs.commands package +===================== + +Submodules +---------- + +cogs.commands.analyze module +---------------------------- + +.. automodule:: cogs.commands.analyze + :members: + :undoc-members: + :show-inheritance: + +cogs.commands.earnings module +----------------------------- + +.. automodule:: cogs.commands.earnings + :members: + :undoc-members: + :show-inheritance: + +cogs.commands.help module +------------------------- + +.. automodule:: cogs.commands.help + :members: + :undoc-members: + :show-inheritance: + +cogs.commands.portfolio module +------------------------------ + +.. automodule:: cogs.commands.portfolio + :members: + :undoc-members: + :show-inheritance: + +cogs.commands.sentiment module +------------------------------ + +.. automodule:: cogs.commands.sentiment + :members: + :undoc-members: + :show-inheritance: + +cogs.commands.stock module +-------------------------- + +.. automodule:: cogs.commands.stock + :members: + :undoc-members: + :show-inheritance: + +Module contents +--------------- + +.. automodule:: cogs.commands + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/cogs.listeners.rst.txt b/_sources/cogs.listeners.rst.txt new file mode 100644 index 00000000..8c70ed1d --- /dev/null +++ b/_sources/cogs.listeners.rst.txt @@ -0,0 +1,29 @@ +cogs.listeners package +====================== + +Submodules +---------- + +cogs.listeners.on\_member\_join module +-------------------------------------- + +.. automodule:: cogs.listeners.on_member_join + :members: + :undoc-members: + :show-inheritance: + +cogs.listeners.on\_raw\_reaction\_add module +-------------------------------------------- + +.. automodule:: cogs.listeners.on_raw_reaction_add + :members: + :undoc-members: + :show-inheritance: + +Module contents +--------------- + +.. automodule:: cogs.listeners + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/cogs.loops.rst.txt b/_sources/cogs.loops.rst.txt new file mode 100644 index 00000000..c9f0cf5b --- /dev/null +++ b/_sources/cogs.loops.rst.txt @@ -0,0 +1,181 @@ +cogs.loops package +================== + +Submodules +---------- + +cogs.loops.assets module +------------------------ + +.. automodule:: cogs.loops.assets + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.earnings\_overview module +------------------------------------ + +.. automodule:: cogs.loops.earnings_overview + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.events module +------------------------ + +.. automodule:: cogs.loops.events + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.funding module +------------------------- + +.. automodule:: cogs.loops.funding + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.gainers module +------------------------- + +.. automodule:: cogs.loops.gainers + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.ideas module +----------------------- + +.. automodule:: cogs.loops.ideas + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.index module +----------------------- + +.. automodule:: cogs.loops.index + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.liquidations module +------------------------------ + +.. automodule:: cogs.loops.liquidations + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.losers module +------------------------ + +.. automodule:: cogs.loops.losers + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.new\_listings module +------------------------------- + +.. automodule:: cogs.loops.new_listings + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.nfts module +---------------------- + +.. automodule:: cogs.loops.nfts + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.option\_alert module +------------------------------- + +.. automodule:: cogs.loops.option_alert + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.options module +------------------------- + +.. automodule:: cogs.loops.options + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.overview module +-------------------------- + +.. automodule:: cogs.loops.overview + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.reddit module +------------------------ + +.. automodule:: cogs.loops.reddit + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.stock\_halts module +------------------------------ + +.. automodule:: cogs.loops.stock_halts + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.stocktwits module +---------------------------- + +.. automodule:: cogs.loops.stocktwits + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.timeline module +-------------------------- + +.. automodule:: cogs.loops.timeline + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.trades module +------------------------ + +.. automodule:: cogs.loops.trades + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.trending module +-------------------------- + +.. automodule:: cogs.loops.trending + :members: + :undoc-members: + :show-inheritance: + +cogs.loops.yield module +----------------------- + +.. automodule:: cogs.loops.yield + :members: + :undoc-members: + :show-inheritance: + +Module contents +--------------- + +.. automodule:: cogs.loops + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/cogs.rst.txt b/_sources/cogs.rst.txt new file mode 100644 index 00000000..94556711 --- /dev/null +++ b/_sources/cogs.rst.txt @@ -0,0 +1,20 @@ +cogs package +============ + +Subpackages +----------- + +.. toctree:: + :maxdepth: 4 + + cogs.commands + cogs.listeners + cogs.loops + +Module contents +--------------- + +.. automodule:: cogs + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/index.rst.txt b/_sources/index.rst.txt new file mode 100644 index 00000000..d3151fc3 --- /dev/null +++ b/_sources/index.rst.txt @@ -0,0 +1,20 @@ +.. FinTwit Bot documentation master file, created by + sphinx-quickstart on Sun Aug 27 20:30:38 2023. + You can adapt this file completely to your liking, but it should at least + contain the root `toctree` directive. + +Welcome to FinTwit Bot's documentation! +======================================= + +.. toctree:: + :maxdepth: 2 + :caption: Contents: + + Modules + +Indices and tables +================== + +* :ref:`genindex` +* :ref:`modindex` +* :ref:`search` diff --git a/_sources/main.rst.txt b/_sources/main.rst.txt new file mode 100644 index 00000000..eace87b8 --- /dev/null +++ b/_sources/main.rst.txt @@ -0,0 +1,7 @@ +main module +=========== + +.. automodule:: main + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/modules.rst.txt b/_sources/modules.rst.txt new file mode 100644 index 00000000..e99e6557 --- /dev/null +++ b/_sources/modules.rst.txt @@ -0,0 +1,13 @@ +src +=== + +.. toctree:: + :maxdepth: 4 + :caption: Contents: + + main + cogs + cogs.commands + cogs.listeners + cogs.loops + util \ No newline at end of file diff --git a/_sources/util.rst.txt b/_sources/util.rst.txt new file mode 100644 index 00000000..b571ab62 --- /dev/null +++ b/_sources/util.rst.txt @@ -0,0 +1,157 @@ +util package +============ + +Submodules +---------- + +util.afterhours module +---------------------- + +.. automodule:: util.afterhours + :members: + :undoc-members: + :show-inheritance: + +util.cg\_data module +-------------------- + +.. automodule:: util.cg_data + :members: + :undoc-members: + :show-inheritance: + +util.confirm\_stock module +-------------------------- + +.. automodule:: util.confirm_stock + :members: + :undoc-members: + :show-inheritance: + +util.db module +-------------- + +.. automodule:: util.db + :members: + :undoc-members: + :show-inheritance: + +util.disc\_util module +---------------------- + +.. automodule:: util.disc_util + :members: + :undoc-members: + :show-inheritance: + +util.earnings\_scraper module +----------------------------- + +.. automodule:: util.earnings_scraper + :members: + :undoc-members: + :show-inheritance: + +util.exchange\_data module +-------------------------- + +.. automodule:: util.exchange_data + :members: + :undoc-members: + :show-inheritance: + +util.formatting module +---------------------- + +.. automodule:: util.formatting + :members: + :undoc-members: + :show-inheritance: + +util.get\_tweet module +---------------------- + +.. automodule:: util.get_tweet + :members: + :undoc-members: + :show-inheritance: + +util.parse\_tweet module +------------------------ + +.. automodule:: util.parse_tweet + :members: + :undoc-members: + :show-inheritance: + +util.sentiment\_analyis module +------------------------------ + +.. automodule:: util.sentiment_analyis + :members: + :undoc-members: + :show-inheritance: + +util.ticker\_classifier module +------------------------------ + +.. automodule:: util.ticker_classifier + :members: + :undoc-members: + :show-inheritance: + +util.trades\_msg module +----------------------- + +.. automodule:: util.trades_msg + :members: + :undoc-members: + :show-inheritance: + +util.tv\_data module +-------------------- + +.. automodule:: util.tv_data + :members: + :undoc-members: + :show-inheritance: + +util.tv\_symbols module +----------------------- + +.. automodule:: util.tv_symbols + :members: + :undoc-members: + :show-inheritance: + +util.tweet\_embed module +------------------------ + +.. automodule:: util.tweet_embed + :members: + :undoc-members: + :show-inheritance: + +util.vars module +---------------- + +.. automodule:: util.vars + :members: + :undoc-members: + :show-inheritance: + +util.yf\_data module +-------------------- + +.. automodule:: util.yf_data + :members: + :undoc-members: + :show-inheritance: + +Module contents +--------------- + +.. automodule:: util + :members: + :undoc-members: + :show-inheritance: diff --git a/_static/basic.css b/_static/basic.css new file mode 100644 index 00000000..2af6139e --- /dev/null +++ b/_static/basic.css @@ -0,0 +1,925 @@ +/* + * basic.css + * ~~~~~~~~~ + * + * Sphinx stylesheet -- basic theme. + * + * :copyright: Copyright 2007-2024 by the Sphinx team, see AUTHORS. + * :license: BSD, see LICENSE for details. + * + */ + +/* -- main layout ----------------------------------------------------------- */ + +div.clearer { + clear: both; +} + +div.section::after { + display: block; + content: ''; + clear: left; +} + +/* -- relbar ---------------------------------------------------------------- */ + +div.related { + width: 100%; + font-size: 90%; +} + +div.related h3 { + display: none; +} + +div.related ul { + margin: 0; + padding: 0 0 0 10px; + list-style: none; +} + +div.related li { + display: inline; +} + +div.related li.right { + float: right; + margin-right: 5px; +} + +/* -- sidebar --------------------------------------------------------------- */ + +div.sphinxsidebarwrapper { + padding: 10px 5px 0 10px; +} + +div.sphinxsidebar { + float: left; + width: 270px; + margin-left: -100%; + font-size: 90%; + word-wrap: break-word; + overflow-wrap : break-word; +} + +div.sphinxsidebar ul { + list-style: none; +} + +div.sphinxsidebar ul ul, +div.sphinxsidebar ul.want-points { + margin-left: 20px; + list-style: square; +} + +div.sphinxsidebar ul ul { + margin-top: 0; + margin-bottom: 0; +} + +div.sphinxsidebar form { + margin-top: 10px; +} + +div.sphinxsidebar input { + border: 1px solid #98dbcc; + font-family: sans-serif; + font-size: 1em; +} + +div.sphinxsidebar #searchbox form.search { + overflow: hidden; +} + +div.sphinxsidebar #searchbox input[type="text"] { + float: left; + width: 80%; + padding: 0.25em; + box-sizing: border-box; +} + +div.sphinxsidebar #searchbox input[type="submit"] { + float: left; + width: 20%; + border-left: none; + padding: 0.25em; + box-sizing: border-box; +} + + +img { + border: 0; + max-width: 100%; +} + +/* -- search page ----------------------------------------------------------- */ + +ul.search { + margin: 10px 0 0 20px; + padding: 0; +} + +ul.search li { + padding: 5px 0 5px 20px; + background-image: url(file.png); + background-repeat: no-repeat; + background-position: 0 7px; +} + +ul.search li a { + font-weight: bold; +} + +ul.search li p.context { + color: #888; + margin: 2px 0 0 30px; + text-align: left; +} + +ul.keywordmatches li.goodmatch a { + font-weight: bold; +} + +/* -- index page ------------------------------------------------------------ */ + +table.contentstable { + width: 90%; + margin-left: auto; + margin-right: auto; +} + +table.contentstable p.biglink { + line-height: 150%; +} + +a.biglink { + font-size: 1.3em; +} + +span.linkdescr { + font-style: italic; + padding-top: 5px; + font-size: 90%; +} + +/* -- general index --------------------------------------------------------- */ + +table.indextable { + width: 100%; +} + +table.indextable td { + text-align: left; + vertical-align: top; +} + +table.indextable ul { + margin-top: 0; + margin-bottom: 0; + list-style-type: none; +} + +table.indextable > tbody > tr > td > ul { + padding-left: 0em; +} + +table.indextable tr.pcap { + height: 10px; +} + +table.indextable tr.cap { + margin-top: 10px; + background-color: #f2f2f2; +} + +img.toggler { + margin-right: 3px; + margin-top: 3px; + cursor: pointer; +} + +div.modindex-jumpbox { + border-top: 1px solid #ddd; + border-bottom: 1px solid #ddd; + margin: 1em 0 1em 0; + padding: 0.4em; +} + +div.genindex-jumpbox { + border-top: 1px solid #ddd; + border-bottom: 1px solid #ddd; + margin: 1em 0 1em 0; + padding: 0.4em; +} + +/* -- domain module index --------------------------------------------------- */ + +table.modindextable td { + padding: 2px; + border-collapse: collapse; +} + +/* -- general body styles --------------------------------------------------- */ + +div.body { + min-width: 360px; + max-width: 800px; +} + +div.body p, div.body dd, div.body li, div.body blockquote { + -moz-hyphens: auto; + -ms-hyphens: auto; + -webkit-hyphens: auto; + hyphens: auto; +} + +a.headerlink { + visibility: hidden; +} + +a:visited { + color: #551A8B; +} + +h1:hover > a.headerlink, +h2:hover > a.headerlink, +h3:hover > a.headerlink, +h4:hover > a.headerlink, +h5:hover > a.headerlink, +h6:hover > a.headerlink, +dt:hover > a.headerlink, +caption:hover > a.headerlink, +p.caption:hover > a.headerlink, +div.code-block-caption:hover > a.headerlink { + visibility: visible; +} + +div.body p.caption { + text-align: inherit; +} + +div.body td { + text-align: left; +} + +.first { + margin-top: 0 !important; +} + +p.rubric { + margin-top: 30px; + font-weight: bold; +} + +img.align-left, figure.align-left, .figure.align-left, object.align-left { + clear: left; + float: left; + margin-right: 1em; +} + +img.align-right, figure.align-right, .figure.align-right, object.align-right { + clear: right; + float: right; + margin-left: 1em; +} + +img.align-center, figure.align-center, .figure.align-center, object.align-center { + display: block; + margin-left: auto; + margin-right: auto; +} + +img.align-default, figure.align-default, .figure.align-default { + display: block; + margin-left: auto; + margin-right: auto; +} + +.align-left { + text-align: left; +} + +.align-center { + text-align: center; +} + +.align-default { + text-align: center; +} + +.align-right { + text-align: right; +} + +/* -- sidebars -------------------------------------------------------------- */ + +div.sidebar, +aside.sidebar { + margin: 0 0 0.5em 1em; + border: 1px solid #ddb; + padding: 7px; + background-color: #ffe; + width: 40%; + float: right; + clear: right; + overflow-x: auto; +} + +p.sidebar-title { + font-weight: bold; +} + +nav.contents, +aside.topic, +div.admonition, div.topic, blockquote { + clear: left; +} + +/* -- topics ---------------------------------------------------------------- */ + +nav.contents, +aside.topic, +div.topic { + border: 1px solid #ccc; + padding: 7px; + margin: 10px 0 10px 0; +} + +p.topic-title { + font-size: 1.1em; + font-weight: bold; + margin-top: 10px; +} + +/* -- admonitions ----------------------------------------------------------- */ + +div.admonition { + margin-top: 10px; + margin-bottom: 10px; + padding: 7px; +} + +div.admonition dt { + font-weight: bold; +} + +p.admonition-title { + margin: 0px 10px 5px 0px; + font-weight: bold; +} + +div.body p.centered { + text-align: center; + margin-top: 25px; +} + +/* -- content of sidebars/topics/admonitions -------------------------------- */ + +div.sidebar > :last-child, +aside.sidebar > :last-child, +nav.contents > :last-child, +aside.topic > :last-child, +div.topic > :last-child, +div.admonition > :last-child { + margin-bottom: 0; +} + +div.sidebar::after, +aside.sidebar::after, +nav.contents::after, +aside.topic::after, +div.topic::after, +div.admonition::after, +blockquote::after { + display: block; + content: ''; + clear: both; +} + +/* -- tables ---------------------------------------------------------------- */ + +table.docutils { + margin-top: 10px; + margin-bottom: 10px; + border: 0; + border-collapse: collapse; +} + +table.align-center { + margin-left: auto; + margin-right: auto; +} + +table.align-default { + margin-left: auto; + margin-right: auto; +} + +table caption span.caption-number { + font-style: italic; +} + +table caption span.caption-text { +} + +table.docutils td, table.docutils th { + padding: 1px 8px 1px 5px; + border-top: 0; + border-left: 0; + border-right: 0; + border-bottom: 1px solid #aaa; +} + +th { + text-align: left; + padding-right: 5px; +} + +table.citation { + border-left: solid 1px gray; + margin-left: 1px; +} + +table.citation td { + border-bottom: none; +} + +th > :first-child, +td > :first-child { + margin-top: 0px; +} + +th > :last-child, +td > :last-child { + margin-bottom: 0px; +} + +/* -- figures --------------------------------------------------------------- */ + +div.figure, figure { + margin: 0.5em; + padding: 0.5em; +} + +div.figure p.caption, figcaption { + padding: 0.3em; +} + +div.figure p.caption span.caption-number, +figcaption span.caption-number { + font-style: italic; +} + +div.figure p.caption span.caption-text, +figcaption span.caption-text { +} + +/* -- field list styles ----------------------------------------------------- */ + +table.field-list td, table.field-list th { + border: 0 !important; +} + +.field-list ul { + margin: 0; + padding-left: 1em; +} + +.field-list p { + margin: 0; +} + +.field-name { + -moz-hyphens: manual; + -ms-hyphens: manual; + -webkit-hyphens: manual; + hyphens: manual; +} + +/* -- hlist styles ---------------------------------------------------------- */ + +table.hlist { + margin: 1em 0; +} + +table.hlist td { + vertical-align: top; +} + +/* -- object description styles --------------------------------------------- */ + +.sig { + font-family: 'Consolas', 'Menlo', 'DejaVu Sans Mono', 'Bitstream Vera Sans Mono', monospace; +} + +.sig-name, code.descname { + background-color: transparent; + font-weight: bold; +} + +.sig-name { + font-size: 1.1em; +} + +code.descname { + font-size: 1.2em; +} + +.sig-prename, code.descclassname { + background-color: transparent; +} + +.optional { + font-size: 1.3em; +} + +.sig-paren { + font-size: larger; +} + +.sig-param.n { + font-style: italic; +} + +/* C++ specific styling */ + +.sig-inline.c-texpr, +.sig-inline.cpp-texpr { + font-family: unset; +} + +.sig.c .k, .sig.c .kt, +.sig.cpp .k, .sig.cpp .kt { + color: #0033B3; +} + +.sig.c .m, +.sig.cpp .m { + color: #1750EB; +} + +.sig.c .s, .sig.c .sc, +.sig.cpp .s, .sig.cpp .sc { + color: #067D17; +} + + +/* -- other body styles ----------------------------------------------------- */ + +ol.arabic { + list-style: decimal; +} + +ol.loweralpha { + list-style: lower-alpha; +} + +ol.upperalpha { + list-style: upper-alpha; +} + +ol.lowerroman { + list-style: lower-roman; +} + +ol.upperroman { + list-style: upper-roman; +} + +:not(li) > ol > li:first-child > :first-child, +:not(li) > ul > li:first-child > :first-child { + margin-top: 0px; +} + +:not(li) > ol > li:last-child > :last-child, +:not(li) > ul > li:last-child > :last-child { + margin-bottom: 0px; +} + +ol.simple ol p, +ol.simple ul p, +ul.simple ol p, +ul.simple ul p { + margin-top: 0; +} + +ol.simple > li:not(:first-child) > p, +ul.simple > li:not(:first-child) > p { + margin-top: 0; +} + +ol.simple p, +ul.simple p { + margin-bottom: 0; +} + +aside.footnote > span, +div.citation > span { + float: left; +} +aside.footnote > span:last-of-type, +div.citation > span:last-of-type { + padding-right: 0.5em; +} +aside.footnote > p { + margin-left: 2em; +} +div.citation > p { + margin-left: 4em; +} +aside.footnote > p:last-of-type, +div.citation > p:last-of-type { + margin-bottom: 0em; +} +aside.footnote > p:last-of-type:after, +div.citation > p:last-of-type:after { + content: ""; + clear: both; +} + +dl.field-list { + display: grid; + grid-template-columns: fit-content(30%) auto; +} + +dl.field-list > dt { + font-weight: bold; + word-break: break-word; + padding-left: 0.5em; + padding-right: 5px; +} + +dl.field-list > dd { + padding-left: 0.5em; + margin-top: 0em; + margin-left: 0em; + margin-bottom: 0em; +} + +dl { + margin-bottom: 15px; +} + +dd > :first-child { + margin-top: 0px; +} + +dd ul, dd table { + margin-bottom: 10px; +} + +dd { + margin-top: 3px; + margin-bottom: 10px; + margin-left: 30px; +} + +.sig dd { + margin-top: 0px; + margin-bottom: 0px; +} + +.sig dl { + margin-top: 0px; + margin-bottom: 0px; +} + +dl > dd:last-child, +dl > dd:last-child > :last-child { + margin-bottom: 0; +} + +dt:target, span.highlighted { + background-color: #fbe54e; +} + +rect.highlighted { + fill: #fbe54e; +} + +dl.glossary dt { + font-weight: bold; + font-size: 1.1em; +} + +.versionmodified { + font-style: italic; +} + +.system-message { + background-color: #fda; + padding: 5px; + border: 3px solid red; +} + +.footnote:target { + background-color: #ffa; +} + +.line-block { + display: block; + margin-top: 1em; + margin-bottom: 1em; +} + +.line-block .line-block { + margin-top: 0; + margin-bottom: 0; + margin-left: 1.5em; +} + +.guilabel, .menuselection { + font-family: sans-serif; +} + +.accelerator { + text-decoration: underline; +} + +.classifier { + font-style: oblique; +} + +.classifier:before { + font-style: normal; + margin: 0 0.5em; + content: ":"; + display: inline-block; +} + +abbr, acronym { + border-bottom: dotted 1px; + cursor: help; +} + +.translated { + background-color: rgba(207, 255, 207, 0.2) +} + +.untranslated { + background-color: rgba(255, 207, 207, 0.2) +} + +/* -- code displays --------------------------------------------------------- */ + +pre { + overflow: auto; + overflow-y: hidden; /* fixes display issues on Chrome browsers */ +} + +pre, div[class*="highlight-"] { + clear: both; +} + +span.pre { + -moz-hyphens: none; + -ms-hyphens: none; + -webkit-hyphens: none; + hyphens: none; + white-space: nowrap; +} + +div[class*="highlight-"] { + margin: 1em 0; +} + +td.linenos pre { + border: 0; + background-color: transparent; + color: #aaa; +} + +table.highlighttable { + display: block; +} + +table.highlighttable tbody { + display: block; +} + +table.highlighttable tr { + display: flex; +} + +table.highlighttable td { + margin: 0; + padding: 0; +} + +table.highlighttable td.linenos { + padding-right: 0.5em; +} + +table.highlighttable td.code { + flex: 1; + overflow: hidden; +} + +.highlight .hll { + display: block; +} + +div.highlight pre, +table.highlighttable pre { + margin: 0; +} + +div.code-block-caption + div { + margin-top: 0; +} + +div.code-block-caption { + margin-top: 1em; + padding: 2px 5px; + font-size: small; +} + +div.code-block-caption code { + background-color: transparent; +} + +table.highlighttable td.linenos, +span.linenos, +div.highlight span.gp { /* gp: Generic.Prompt */ + user-select: none; + -webkit-user-select: text; /* Safari fallback only */ + -webkit-user-select: none; /* Chrome/Safari */ + -moz-user-select: none; /* Firefox */ + -ms-user-select: none; /* IE10+ */ +} + +div.code-block-caption span.caption-number { + padding: 0.1em 0.3em; + font-style: italic; +} + +div.code-block-caption span.caption-text { +} + +div.literal-block-wrapper { + margin: 1em 0; +} + +code.xref, a code { + background-color: transparent; + font-weight: bold; +} + +h1 code, h2 code, h3 code, h4 code, h5 code, h6 code { + background-color: transparent; +} + +.viewcode-link { + float: right; +} + +.viewcode-back { + float: right; + font-family: sans-serif; +} + +div.viewcode-block:target { + margin: -1px -10px; + padding: 0 10px; +} + +/* -- math display ---------------------------------------------------------- */ + +img.math { + vertical-align: middle; +} + +div.body div.math p { + text-align: center; +} + +span.eqno { + float: right; +} + +span.eqno a.headerlink { + position: absolute; + z-index: 1; +} + +div.math:hover a.headerlink { + visibility: visible; +} + +/* -- printout stylesheet --------------------------------------------------- */ + +@media print { + div.document, + div.documentwrapper, + div.bodywrapper { + margin: 0 !important; + width: 100%; + } + + div.sphinxsidebar, + div.related, + div.footer, + #top-link { + display: none; + } +} \ No newline at end of file diff --git a/_static/doctools.js b/_static/doctools.js new file mode 100644 index 00000000..4d67807d --- /dev/null +++ b/_static/doctools.js @@ -0,0 +1,156 @@ +/* + * doctools.js + * ~~~~~~~~~~~ + * + * Base JavaScript utilities for all Sphinx HTML documentation. + * + * :copyright: Copyright 2007-2024 by the Sphinx team, see AUTHORS. + * :license: BSD, see LICENSE for details. + * + */ +"use strict"; + +const BLACKLISTED_KEY_CONTROL_ELEMENTS = new Set([ + "TEXTAREA", + "INPUT", + "SELECT", + "BUTTON", +]); + +const _ready = (callback) => { + if (document.readyState !== "loading") { + callback(); + } else { + document.addEventListener("DOMContentLoaded", callback); + } +}; + +/** + * Small JavaScript module for the documentation. + */ +const Documentation = { + init: () => { + Documentation.initDomainIndexTable(); + Documentation.initOnKeyListeners(); + }, + + /** + * i18n support + */ + TRANSLATIONS: {}, + PLURAL_EXPR: (n) => (n === 1 ? 0 : 1), + LOCALE: "unknown", + + // gettext and ngettext don't access this so that the functions + // can safely bound to a different name (_ = Documentation.gettext) + gettext: (string) => { + const translated = Documentation.TRANSLATIONS[string]; + switch (typeof translated) { + case "undefined": + return string; // no translation + case "string": + return translated; // translation exists + default: + return translated[0]; // (singular, plural) translation tuple exists + } + }, + + ngettext: (singular, plural, n) => { + const translated = Documentation.TRANSLATIONS[singular]; + if (typeof translated !== "undefined") + return translated[Documentation.PLURAL_EXPR(n)]; + return n === 1 ? singular : plural; + }, + + addTranslations: (catalog) => { + Object.assign(Documentation.TRANSLATIONS, catalog.messages); + Documentation.PLURAL_EXPR = new Function( + "n", + `return (${catalog.plural_expr})` + ); + Documentation.LOCALE = catalog.locale; + }, + + /** + * helper function to focus on search bar + */ + focusSearchBar: () => { + document.querySelectorAll("input[name=q]")[0]?.focus(); + }, + + /** + * Initialise the domain index toggle buttons + */ + initDomainIndexTable: () => { + const toggler = (el) => { + const idNumber = el.id.substr(7); + const toggledRows = document.querySelectorAll(`tr.cg-${idNumber}`); + if (el.src.substr(-9) === "minus.png") { + el.src = `${el.src.substr(0, el.src.length - 9)}plus.png`; + toggledRows.forEach((el) => (el.style.display = "none")); + } else { + el.src = `${el.src.substr(0, el.src.length - 8)}minus.png`; + toggledRows.forEach((el) => (el.style.display = "")); + } + }; + + const togglerElements = document.querySelectorAll("img.toggler"); + togglerElements.forEach((el) => + el.addEventListener("click", (event) => toggler(event.currentTarget)) + ); + togglerElements.forEach((el) => (el.style.display = "")); + if (DOCUMENTATION_OPTIONS.COLLAPSE_INDEX) togglerElements.forEach(toggler); + }, + + initOnKeyListeners: () => { + // only install a listener if it is really needed + if ( + !DOCUMENTATION_OPTIONS.NAVIGATION_WITH_KEYS && + !DOCUMENTATION_OPTIONS.ENABLE_SEARCH_SHORTCUTS + ) + return; + + document.addEventListener("keydown", (event) => { + // bail for input elements + if (BLACKLISTED_KEY_CONTROL_ELEMENTS.has(document.activeElement.tagName)) return; + // bail with special keys + if (event.altKey || event.ctrlKey || event.metaKey) return; + + if (!event.shiftKey) { + switch (event.key) { + case "ArrowLeft": + if (!DOCUMENTATION_OPTIONS.NAVIGATION_WITH_KEYS) break; + + const prevLink = document.querySelector('link[rel="prev"]'); + if (prevLink && prevLink.href) { + window.location.href = prevLink.href; + event.preventDefault(); + } + break; + case "ArrowRight": + if (!DOCUMENTATION_OPTIONS.NAVIGATION_WITH_KEYS) break; + + const nextLink = document.querySelector('link[rel="next"]'); + if (nextLink && nextLink.href) { + window.location.href = nextLink.href; + event.preventDefault(); + } + break; + } + } + + // some keyboard layouts may need Shift to get / + switch (event.key) { + case "/": + if (!DOCUMENTATION_OPTIONS.ENABLE_SEARCH_SHORTCUTS) break; + Documentation.focusSearchBar(); + event.preventDefault(); + } + }); + }, +}; + +// quick alias for translations +const _ = Documentation.gettext; + +_ready(Documentation.init); diff --git a/_static/documentation_options.js b/_static/documentation_options.js new file mode 100644 index 00000000..3f423940 --- /dev/null +++ b/_static/documentation_options.js @@ -0,0 +1,13 @@ +const DOCUMENTATION_OPTIONS = { + VERSION: '1.2.0', + LANGUAGE: 'en', + COLLAPSE_INDEX: false, + BUILDER: 'html', + FILE_SUFFIX: '.html', + LINK_SUFFIX: '.html', + HAS_SOURCE: true, + SOURCELINK_SUFFIX: '.txt', + NAVIGATION_WITH_KEYS: false, + SHOW_SEARCH_SUMMARY: true, + ENABLE_SEARCH_SHORTCUTS: true, +}; \ No newline at end of file diff --git a/_static/file.png b/_static/file.png new file mode 100644 index 00000000..a858a410 Binary files /dev/null and b/_static/file.png differ diff --git a/_static/fintwit-nobg.ico b/_static/fintwit-nobg.ico new file mode 100644 index 00000000..23e89b01 Binary files /dev/null and b/_static/fintwit-nobg.ico differ diff --git a/_static/fintwit.png b/_static/fintwit.png new file mode 100644 index 00000000..bf48d018 Binary files /dev/null and b/_static/fintwit.png differ diff --git a/_static/graphviz.css b/_static/graphviz.css new file mode 100644 index 00000000..027576e3 --- /dev/null +++ b/_static/graphviz.css @@ -0,0 +1,19 @@ +/* + * graphviz.css + * ~~~~~~~~~~~~ + * + * Sphinx stylesheet -- graphviz extension. + * + * :copyright: Copyright 2007-2024 by the Sphinx team, see AUTHORS. + * :license: BSD, see LICENSE for details. + * + */ + +img.graphviz { + border: 0; + max-width: 100%; +} + +object.graphviz { + max-width: 100%; +} diff --git a/_static/language_data.js b/_static/language_data.js new file mode 100644 index 00000000..367b8ed8 --- /dev/null +++ b/_static/language_data.js @@ -0,0 +1,199 @@ +/* + * language_data.js + * ~~~~~~~~~~~~~~~~ + * + * This script contains the language-specific data used by searchtools.js, + * namely the list of stopwords, stemmer, scorer and splitter. + * + * :copyright: Copyright 2007-2024 by the Sphinx team, see AUTHORS. + * :license: BSD, see LICENSE for details. + * + */ + +var stopwords = ["a", "and", "are", "as", "at", "be", "but", "by", "for", "if", "in", "into", "is", "it", "near", "no", "not", "of", "on", "or", "such", "that", "the", "their", "then", "there", "these", "they", "this", "to", "was", "will", "with"]; + + +/* Non-minified version is copied as a separate JS file, if available */ + +/** + * Porter Stemmer + */ +var Stemmer = function() { + + var step2list = { + ational: 'ate', + tional: 'tion', + enci: 'ence', + anci: 'ance', + izer: 'ize', + bli: 'ble', + alli: 'al', + entli: 'ent', + eli: 'e', + ousli: 'ous', + ization: 'ize', + ation: 'ate', + ator: 'ate', + alism: 'al', + iveness: 'ive', + fulness: 'ful', + ousness: 'ous', + aliti: 'al', + iviti: 'ive', + biliti: 'ble', + logi: 'log' + }; + + var step3list = { + icate: 'ic', + ative: '', + alize: 'al', + iciti: 'ic', + ical: 'ic', + ful: '', + ness: '' + }; + + var c = "[^aeiou]"; // consonant + var v = "[aeiouy]"; // vowel + var C = c + "[^aeiouy]*"; // consonant sequence + var V = v + "[aeiou]*"; // vowel sequence + + var mgr0 = "^(" + C + ")?" + V + C; // [C]VC... is m>0 + var meq1 = "^(" + C + ")?" + V + C + "(" + V + ")?$"; // [C]VC[V] is m=1 + var mgr1 = "^(" + C + ")?" + V + C + V + C; // [C]VCVC... is m>1 + var s_v = "^(" + C + ")?" + v; // vowel in stem + + this.stemWord = function (w) { + var stem; + var suffix; + var firstch; + var origword = w; + + if (w.length < 3) + return w; + + var re; + var re2; + var re3; + var re4; + + firstch = w.substr(0,1); + if (firstch == "y") + w = firstch.toUpperCase() + w.substr(1); + + // Step 1a + re = /^(.+?)(ss|i)es$/; + re2 = /^(.+?)([^s])s$/; + + if (re.test(w)) + w = w.replace(re,"$1$2"); + else if (re2.test(w)) + w = w.replace(re2,"$1$2"); + + // Step 1b + re = /^(.+?)eed$/; + re2 = /^(.+?)(ed|ing)$/; + if (re.test(w)) { + var fp = re.exec(w); + re = new RegExp(mgr0); + if (re.test(fp[1])) { + re = /.$/; + w = w.replace(re,""); + } + } + else if (re2.test(w)) { + var fp = re2.exec(w); + stem = fp[1]; + re2 = new RegExp(s_v); + if (re2.test(stem)) { + w = stem; + re2 = /(at|bl|iz)$/; + re3 = new RegExp("([^aeiouylsz])\\1$"); + re4 = new RegExp("^" + C + v + "[^aeiouwxy]$"); + if (re2.test(w)) + w = w + "e"; + else if (re3.test(w)) { + re = /.$/; + w = w.replace(re,""); + } + else if (re4.test(w)) + w = w + "e"; + } + } + + // Step 1c + re = /^(.+?)y$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + re = new RegExp(s_v); + if (re.test(stem)) + w = stem + "i"; + } + + // Step 2 + re = /^(.+?)(ational|tional|enci|anci|izer|bli|alli|entli|eli|ousli|ization|ation|ator|alism|iveness|fulness|ousness|aliti|iviti|biliti|logi)$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + suffix = fp[2]; + re = new RegExp(mgr0); + if (re.test(stem)) + w = stem + step2list[suffix]; + } + + // Step 3 + re = /^(.+?)(icate|ative|alize|iciti|ical|ful|ness)$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + suffix = fp[2]; + re = new RegExp(mgr0); + if (re.test(stem)) + w = stem + step3list[suffix]; + } + + // Step 4 + re = /^(.+?)(al|ance|ence|er|ic|able|ible|ant|ement|ment|ent|ou|ism|ate|iti|ous|ive|ize)$/; + re2 = /^(.+?)(s|t)(ion)$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + re = new RegExp(mgr1); + if (re.test(stem)) + w = stem; + } + else if (re2.test(w)) { + var fp = re2.exec(w); + stem = fp[1] + fp[2]; + re2 = new RegExp(mgr1); + if (re2.test(stem)) + w = stem; + } + + // Step 5 + re = /^(.+?)e$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + re = new RegExp(mgr1); + re2 = new RegExp(meq1); + re3 = new RegExp("^" + C + v + "[^aeiouwxy]$"); + if (re.test(stem) || (re2.test(stem) && !(re3.test(stem)))) + w = stem; + } + re = /ll$/; + re2 = new RegExp(mgr1); + if (re.test(w) && re2.test(w)) { + re = /.$/; + w = w.replace(re,""); + } + + // and turn initial Y back to y + if (firstch == "y") + w = firstch.toLowerCase() + w.substr(1); + return w; + } +} + diff --git a/_static/minus.png b/_static/minus.png new file mode 100644 index 00000000..d96755fd Binary files /dev/null and b/_static/minus.png differ diff --git a/_static/plus.png b/_static/plus.png new file mode 100644 index 00000000..7107cec9 Binary files /dev/null and b/_static/plus.png differ diff --git a/_static/pygments.css b/_static/pygments.css new file mode 100644 index 00000000..012e6a00 --- /dev/null +++ b/_static/pygments.css @@ -0,0 +1,152 @@ +html[data-theme="light"] .highlight pre { line-height: 125%; } +html[data-theme="light"] .highlight td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +html[data-theme="light"] .highlight span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +html[data-theme="light"] .highlight td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +html[data-theme="light"] .highlight span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +html[data-theme="light"] .highlight .hll { background-color: #fae4c2 } +html[data-theme="light"] .highlight { background: #fefefe; color: #080808 } +html[data-theme="light"] .highlight .c { color: #515151 } /* Comment */ +html[data-theme="light"] .highlight .err { color: #a12236 } /* Error */ +html[data-theme="light"] .highlight .k { color: #6730c5 } /* Keyword */ +html[data-theme="light"] .highlight .l { color: #7f4707 } /* Literal */ +html[data-theme="light"] .highlight .n { color: #080808 } /* Name */ +html[data-theme="light"] .highlight .o { color: #00622f } /* Operator */ +html[data-theme="light"] .highlight .p { color: #080808 } /* Punctuation */ +html[data-theme="light"] .highlight .ch { color: #515151 } /* Comment.Hashbang */ +html[data-theme="light"] .highlight .cm { color: #515151 } /* Comment.Multiline */ +html[data-theme="light"] .highlight .cp { color: #515151 } /* Comment.Preproc */ +html[data-theme="light"] .highlight .cpf { color: #515151 } /* Comment.PreprocFile */ +html[data-theme="light"] .highlight .c1 { color: #515151 } /* Comment.Single */ +html[data-theme="light"] .highlight .cs { color: #515151 } /* Comment.Special */ +html[data-theme="light"] .highlight .gd { color: #005b82 } /* Generic.Deleted */ +html[data-theme="light"] .highlight .ge { font-style: italic } /* Generic.Emph */ +html[data-theme="light"] .highlight .gh { color: #005b82 } /* Generic.Heading */ +html[data-theme="light"] .highlight .gs { font-weight: bold } /* Generic.Strong */ +html[data-theme="light"] .highlight .gu { color: #005b82 } /* Generic.Subheading */ +html[data-theme="light"] .highlight .kc { color: #6730c5 } /* Keyword.Constant */ +html[data-theme="light"] .highlight .kd { color: #6730c5 } /* Keyword.Declaration */ +html[data-theme="light"] .highlight .kn { color: #6730c5 } /* Keyword.Namespace */ +html[data-theme="light"] .highlight .kp { color: #6730c5 } /* Keyword.Pseudo */ +html[data-theme="light"] .highlight .kr { color: #6730c5 } /* Keyword.Reserved */ +html[data-theme="light"] .highlight .kt { color: #7f4707 } /* Keyword.Type */ +html[data-theme="light"] .highlight .ld { color: #7f4707 } /* Literal.Date */ +html[data-theme="light"] .highlight .m { color: #7f4707 } /* Literal.Number */ +html[data-theme="light"] .highlight .s { color: #00622f } /* Literal.String */ +html[data-theme="light"] .highlight .na { color: #912583 } /* Name.Attribute */ +html[data-theme="light"] .highlight .nb { color: #7f4707 } /* Name.Builtin */ +html[data-theme="light"] .highlight .nc { color: #005b82 } /* Name.Class */ +html[data-theme="light"] .highlight .no { color: #005b82 } /* Name.Constant */ +html[data-theme="light"] .highlight .nd { color: #7f4707 } /* Name.Decorator */ +html[data-theme="light"] .highlight .ni { color: #00622f } /* Name.Entity */ +html[data-theme="light"] .highlight .ne { color: #6730c5 } /* Name.Exception */ +html[data-theme="light"] .highlight .nf { color: #005b82 } /* Name.Function */ +html[data-theme="light"] .highlight .nl { color: #7f4707 } /* Name.Label */ +html[data-theme="light"] .highlight .nn { color: #080808 } /* Name.Namespace */ +html[data-theme="light"] .highlight .nx { color: #080808 } /* Name.Other */ +html[data-theme="light"] .highlight .py { color: #005b82 } /* Name.Property */ +html[data-theme="light"] .highlight .nt { color: #005b82 } /* Name.Tag */ +html[data-theme="light"] .highlight .nv { color: #a12236 } /* Name.Variable */ +html[data-theme="light"] .highlight .ow { color: #6730c5 } /* Operator.Word */ +html[data-theme="light"] .highlight .pm { color: #080808 } /* Punctuation.Marker */ +html[data-theme="light"] .highlight .w { color: #080808 } /* Text.Whitespace */ +html[data-theme="light"] .highlight .mb { color: #7f4707 } /* Literal.Number.Bin */ +html[data-theme="light"] .highlight .mf { color: #7f4707 } /* Literal.Number.Float */ +html[data-theme="light"] .highlight .mh { color: #7f4707 } /* Literal.Number.Hex */ +html[data-theme="light"] .highlight .mi { color: #7f4707 } /* Literal.Number.Integer */ +html[data-theme="light"] .highlight .mo { color: #7f4707 } /* Literal.Number.Oct */ +html[data-theme="light"] .highlight .sa { color: #00622f } /* Literal.String.Affix */ +html[data-theme="light"] .highlight .sb { color: #00622f } /* Literal.String.Backtick */ +html[data-theme="light"] .highlight .sc { color: #00622f } /* Literal.String.Char */ +html[data-theme="light"] .highlight .dl { color: #00622f } /* Literal.String.Delimiter */ +html[data-theme="light"] .highlight .sd { color: #00622f } /* Literal.String.Doc */ +html[data-theme="light"] .highlight .s2 { color: #00622f } /* Literal.String.Double */ +html[data-theme="light"] .highlight .se { color: #00622f } /* Literal.String.Escape */ +html[data-theme="light"] .highlight .sh { color: #00622f } /* Literal.String.Heredoc */ +html[data-theme="light"] .highlight .si { color: #00622f } /* Literal.String.Interpol */ +html[data-theme="light"] .highlight .sx { color: #00622f } /* Literal.String.Other */ +html[data-theme="light"] .highlight .sr { color: #a12236 } /* Literal.String.Regex */ +html[data-theme="light"] .highlight .s1 { color: #00622f } /* Literal.String.Single */ +html[data-theme="light"] .highlight .ss { color: #005b82 } /* Literal.String.Symbol */ +html[data-theme="light"] .highlight .bp { color: #7f4707 } /* Name.Builtin.Pseudo */ +html[data-theme="light"] .highlight .fm { color: #005b82 } /* Name.Function.Magic */ +html[data-theme="light"] .highlight .vc { color: #a12236 } /* Name.Variable.Class */ +html[data-theme="light"] .highlight .vg { color: #a12236 } /* Name.Variable.Global */ +html[data-theme="light"] .highlight .vi { color: #a12236 } /* Name.Variable.Instance */ +html[data-theme="light"] .highlight .vm { color: #7f4707 } /* Name.Variable.Magic */ +html[data-theme="light"] .highlight .il { color: #7f4707 } /* Literal.Number.Integer.Long */ +html[data-theme="dark"] .highlight pre { line-height: 125%; } +html[data-theme="dark"] .highlight td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +html[data-theme="dark"] .highlight span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +html[data-theme="dark"] .highlight td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +html[data-theme="dark"] .highlight span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +html[data-theme="dark"] .highlight .hll { background-color: #ffd9002e } +html[data-theme="dark"] .highlight { background: #2b2b2b; color: #f8f8f2 } +html[data-theme="dark"] .highlight .c { color: #ffd900 } /* Comment */ +html[data-theme="dark"] .highlight .err { color: #ffa07a } /* Error */ +html[data-theme="dark"] .highlight .k { color: #dcc6e0 } /* Keyword */ +html[data-theme="dark"] .highlight .l { color: #ffd900 } /* Literal */ +html[data-theme="dark"] .highlight .n { color: #f8f8f2 } /* Name */ +html[data-theme="dark"] .highlight .o { color: #abe338 } /* Operator */ +html[data-theme="dark"] .highlight .p { color: #f8f8f2 } /* Punctuation */ +html[data-theme="dark"] .highlight .ch { color: #ffd900 } /* Comment.Hashbang */ +html[data-theme="dark"] .highlight .cm { color: #ffd900 } /* Comment.Multiline */ +html[data-theme="dark"] .highlight .cp { color: #ffd900 } /* Comment.Preproc */ +html[data-theme="dark"] .highlight .cpf { color: #ffd900 } /* Comment.PreprocFile */ +html[data-theme="dark"] .highlight .c1 { color: #ffd900 } /* Comment.Single */ +html[data-theme="dark"] .highlight .cs { color: #ffd900 } /* Comment.Special */ +html[data-theme="dark"] .highlight .gd { color: #00e0e0 } /* Generic.Deleted */ +html[data-theme="dark"] .highlight .ge { font-style: italic } /* Generic.Emph */ +html[data-theme="dark"] .highlight .gh { color: #00e0e0 } /* Generic.Heading */ +html[data-theme="dark"] .highlight .gs { font-weight: bold } /* Generic.Strong */ +html[data-theme="dark"] .highlight .gu { color: #00e0e0 } /* Generic.Subheading */ +html[data-theme="dark"] .highlight .kc { color: #dcc6e0 } /* Keyword.Constant */ +html[data-theme="dark"] .highlight .kd { color: #dcc6e0 } /* Keyword.Declaration */ +html[data-theme="dark"] .highlight .kn { color: #dcc6e0 } /* Keyword.Namespace */ +html[data-theme="dark"] .highlight .kp { color: #dcc6e0 } /* Keyword.Pseudo */ +html[data-theme="dark"] .highlight .kr { color: #dcc6e0 } /* Keyword.Reserved */ +html[data-theme="dark"] .highlight .kt { color: #ffd900 } /* Keyword.Type */ +html[data-theme="dark"] .highlight .ld { color: #ffd900 } /* Literal.Date */ +html[data-theme="dark"] .highlight .m { color: #ffd900 } /* Literal.Number */ +html[data-theme="dark"] .highlight .s { color: #abe338 } /* Literal.String */ +html[data-theme="dark"] .highlight .na { color: #ffd900 } /* Name.Attribute */ +html[data-theme="dark"] .highlight .nb { color: #ffd900 } /* Name.Builtin */ +html[data-theme="dark"] .highlight .nc { color: #00e0e0 } /* Name.Class */ +html[data-theme="dark"] .highlight .no { color: #00e0e0 } /* Name.Constant */ +html[data-theme="dark"] .highlight .nd { color: #ffd900 } /* Name.Decorator */ +html[data-theme="dark"] .highlight .ni { color: #abe338 } /* Name.Entity */ +html[data-theme="dark"] .highlight .ne { color: #dcc6e0 } /* Name.Exception */ +html[data-theme="dark"] .highlight .nf { color: #00e0e0 } /* Name.Function */ +html[data-theme="dark"] .highlight .nl { color: #ffd900 } /* Name.Label */ +html[data-theme="dark"] .highlight .nn { color: #f8f8f2 } /* Name.Namespace */ +html[data-theme="dark"] .highlight .nx { color: #f8f8f2 } /* Name.Other */ +html[data-theme="dark"] .highlight .py { color: #00e0e0 } /* Name.Property */ +html[data-theme="dark"] .highlight .nt { color: #00e0e0 } /* Name.Tag */ +html[data-theme="dark"] .highlight .nv { color: #ffa07a } /* Name.Variable */ +html[data-theme="dark"] .highlight .ow { color: #dcc6e0 } /* Operator.Word */ +html[data-theme="dark"] .highlight .pm { color: #f8f8f2 } /* Punctuation.Marker */ +html[data-theme="dark"] .highlight .w { color: #f8f8f2 } /* Text.Whitespace */ +html[data-theme="dark"] .highlight .mb { color: #ffd900 } /* Literal.Number.Bin */ +html[data-theme="dark"] .highlight .mf { color: #ffd900 } /* Literal.Number.Float */ +html[data-theme="dark"] .highlight .mh { color: #ffd900 } /* Literal.Number.Hex */ +html[data-theme="dark"] .highlight .mi { color: #ffd900 } /* Literal.Number.Integer */ +html[data-theme="dark"] .highlight .mo { color: #ffd900 } /* Literal.Number.Oct */ +html[data-theme="dark"] .highlight .sa { color: #abe338 } /* Literal.String.Affix */ +html[data-theme="dark"] .highlight .sb { color: #abe338 } /* Literal.String.Backtick */ +html[data-theme="dark"] .highlight .sc { color: #abe338 } /* Literal.String.Char */ +html[data-theme="dark"] .highlight .dl { color: #abe338 } /* Literal.String.Delimiter */ +html[data-theme="dark"] .highlight .sd { color: #abe338 } /* Literal.String.Doc */ +html[data-theme="dark"] .highlight .s2 { color: #abe338 } /* Literal.String.Double */ +html[data-theme="dark"] .highlight .se { color: #abe338 } /* Literal.String.Escape */ +html[data-theme="dark"] .highlight .sh { color: #abe338 } /* Literal.String.Heredoc */ +html[data-theme="dark"] .highlight .si { color: #abe338 } /* Literal.String.Interpol */ +html[data-theme="dark"] .highlight .sx { color: #abe338 } /* Literal.String.Other */ +html[data-theme="dark"] .highlight .sr { color: #ffa07a } /* Literal.String.Regex */ +html[data-theme="dark"] .highlight .s1 { color: #abe338 } /* Literal.String.Single */ +html[data-theme="dark"] .highlight .ss { color: #00e0e0 } /* Literal.String.Symbol */ +html[data-theme="dark"] .highlight .bp { color: #ffd900 } /* Name.Builtin.Pseudo */ +html[data-theme="dark"] .highlight .fm { color: #00e0e0 } /* Name.Function.Magic */ +html[data-theme="dark"] .highlight .vc { color: #ffa07a } /* Name.Variable.Class */ +html[data-theme="dark"] .highlight .vg { color: #ffa07a } /* Name.Variable.Global */ +html[data-theme="dark"] .highlight .vi { color: #ffa07a } /* Name.Variable.Instance */ +html[data-theme="dark"] .highlight .vm { color: #ffd900 } /* Name.Variable.Magic */ +html[data-theme="dark"] .highlight .il { color: #ffd900 } /* Literal.Number.Integer.Long */ \ No newline at end of file diff --git a/_static/scripts/bootstrap.js b/_static/scripts/bootstrap.js new file mode 100644 index 00000000..c8178deb --- /dev/null +++ b/_static/scripts/bootstrap.js @@ -0,0 +1,3 @@ +/*! For license information please see bootstrap.js.LICENSE.txt */ +(()=>{"use strict";var t={d:(e,i)=>{for(var n in i)t.o(i,n)&&!t.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:i[n]})},o:(t,e)=>Object.prototype.hasOwnProperty.call(t,e),r:t=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(t,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(t,"__esModule",{value:!0})}},e={};t.r(e),t.d(e,{afterMain:()=>E,afterRead:()=>v,afterWrite:()=>C,applyStyles:()=>$,arrow:()=>J,auto:()=>a,basePlacements:()=>l,beforeMain:()=>y,beforeRead:()=>_,beforeWrite:()=>A,bottom:()=>s,clippingParents:()=>d,computeStyles:()=>it,createPopper:()=>Dt,createPopperBase:()=>St,createPopperLite:()=>$t,detectOverflow:()=>_t,end:()=>h,eventListeners:()=>st,flip:()=>bt,hide:()=>wt,left:()=>r,main:()=>w,modifierPhases:()=>O,offset:()=>Et,placements:()=>g,popper:()=>f,popperGenerator:()=>Lt,popperOffsets:()=>At,preventOverflow:()=>Tt,read:()=>b,reference:()=>p,right:()=>o,start:()=>c,top:()=>n,variationPlacements:()=>m,viewport:()=>u,write:()=>T});var i={};t.r(i),t.d(i,{Alert:()=>Oe,Button:()=>ke,Carousel:()=>li,Collapse:()=>Ei,Dropdown:()=>Ki,Modal:()=>Ln,Offcanvas:()=>Kn,Popover:()=>bs,ScrollSpy:()=>Ls,Tab:()=>Js,Toast:()=>po,Tooltip:()=>fs});var n="top",s="bottom",o="right",r="left",a="auto",l=[n,s,o,r],c="start",h="end",d="clippingParents",u="viewport",f="popper",p="reference",m=l.reduce((function(t,e){return t.concat([e+"-"+c,e+"-"+h])}),[]),g=[].concat(l,[a]).reduce((function(t,e){return t.concat([e,e+"-"+c,e+"-"+h])}),[]),_="beforeRead",b="read",v="afterRead",y="beforeMain",w="main",E="afterMain",A="beforeWrite",T="write",C="afterWrite",O=[_,b,v,y,w,E,A,T,C];function x(t){return t?(t.nodeName||"").toLowerCase():null}function k(t){if(null==t)return window;if("[object Window]"!==t.toString()){var e=t.ownerDocument;return e&&e.defaultView||window}return t}function L(t){return t instanceof k(t).Element||t instanceof Element}function S(t){return t instanceof k(t).HTMLElement||t instanceof HTMLElement}function D(t){return"undefined"!=typeof ShadowRoot&&(t instanceof k(t).ShadowRoot||t instanceof ShadowRoot)}const $={name:"applyStyles",enabled:!0,phase:"write",fn:function(t){var e=t.state;Object.keys(e.elements).forEach((function(t){var i=e.styles[t]||{},n=e.attributes[t]||{},s=e.elements[t];S(s)&&x(s)&&(Object.assign(s.style,i),Object.keys(n).forEach((function(t){var e=n[t];!1===e?s.removeAttribute(t):s.setAttribute(t,!0===e?"":e)})))}))},effect:function(t){var e=t.state,i={popper:{position:e.options.strategy,left:"0",top:"0",margin:"0"},arrow:{position:"absolute"},reference:{}};return Object.assign(e.elements.popper.style,i.popper),e.styles=i,e.elements.arrow&&Object.assign(e.elements.arrow.style,i.arrow),function(){Object.keys(e.elements).forEach((function(t){var n=e.elements[t],s=e.attributes[t]||{},o=Object.keys(e.styles.hasOwnProperty(t)?e.styles[t]:i[t]).reduce((function(t,e){return t[e]="",t}),{});S(n)&&x(n)&&(Object.assign(n.style,o),Object.keys(s).forEach((function(t){n.removeAttribute(t)})))}))}},requires:["computeStyles"]};function I(t){return t.split("-")[0]}var N=Math.max,P=Math.min,M=Math.round;function j(){var t=navigator.userAgentData;return null!=t&&t.brands&&Array.isArray(t.brands)?t.brands.map((function(t){return t.brand+"/"+t.version})).join(" "):navigator.userAgent}function F(){return!/^((?!chrome|android).)*safari/i.test(j())}function H(t,e,i){void 0===e&&(e=!1),void 0===i&&(i=!1);var n=t.getBoundingClientRect(),s=1,o=1;e&&S(t)&&(s=t.offsetWidth>0&&M(n.width)/t.offsetWidth||1,o=t.offsetHeight>0&&M(n.height)/t.offsetHeight||1);var r=(L(t)?k(t):window).visualViewport,a=!F()&&i,l=(n.left+(a&&r?r.offsetLeft:0))/s,c=(n.top+(a&&r?r.offsetTop:0))/o,h=n.width/s,d=n.height/o;return{width:h,height:d,top:c,right:l+h,bottom:c+d,left:l,x:l,y:c}}function B(t){var e=H(t),i=t.offsetWidth,n=t.offsetHeight;return Math.abs(e.width-i)<=1&&(i=e.width),Math.abs(e.height-n)<=1&&(n=e.height),{x:t.offsetLeft,y:t.offsetTop,width:i,height:n}}function W(t,e){var i=e.getRootNode&&e.getRootNode();if(t.contains(e))return!0;if(i&&D(i)){var n=e;do{if(n&&t.isSameNode(n))return!0;n=n.parentNode||n.host}while(n)}return!1}function z(t){return k(t).getComputedStyle(t)}function R(t){return["table","td","th"].indexOf(x(t))>=0}function q(t){return((L(t)?t.ownerDocument:t.document)||window.document).documentElement}function V(t){return"html"===x(t)?t:t.assignedSlot||t.parentNode||(D(t)?t.host:null)||q(t)}function Y(t){return S(t)&&"fixed"!==z(t).position?t.offsetParent:null}function K(t){for(var e=k(t),i=Y(t);i&&R(i)&&"static"===z(i).position;)i=Y(i);return i&&("html"===x(i)||"body"===x(i)&&"static"===z(i).position)?e:i||function(t){var e=/firefox/i.test(j());if(/Trident/i.test(j())&&S(t)&&"fixed"===z(t).position)return null;var i=V(t);for(D(i)&&(i=i.host);S(i)&&["html","body"].indexOf(x(i))<0;){var n=z(i);if("none"!==n.transform||"none"!==n.perspective||"paint"===n.contain||-1!==["transform","perspective"].indexOf(n.willChange)||e&&"filter"===n.willChange||e&&n.filter&&"none"!==n.filter)return i;i=i.parentNode}return null}(t)||e}function Q(t){return["top","bottom"].indexOf(t)>=0?"x":"y"}function X(t,e,i){return N(t,P(e,i))}function U(t){return Object.assign({},{top:0,right:0,bottom:0,left:0},t)}function G(t,e){return e.reduce((function(e,i){return e[i]=t,e}),{})}const J={name:"arrow",enabled:!0,phase:"main",fn:function(t){var e,i=t.state,a=t.name,c=t.options,h=i.elements.arrow,d=i.modifiersData.popperOffsets,u=I(i.placement),f=Q(u),p=[r,o].indexOf(u)>=0?"height":"width";if(h&&d){var m=function(t,e){return U("number"!=typeof(t="function"==typeof t?t(Object.assign({},e.rects,{placement:e.placement})):t)?t:G(t,l))}(c.padding,i),g=B(h),_="y"===f?n:r,b="y"===f?s:o,v=i.rects.reference[p]+i.rects.reference[f]-d[f]-i.rects.popper[p],y=d[f]-i.rects.reference[f],w=K(h),E=w?"y"===f?w.clientHeight||0:w.clientWidth||0:0,A=v/2-y/2,T=m[_],C=E-g[p]-m[b],O=E/2-g[p]/2+A,x=X(T,O,C),k=f;i.modifiersData[a]=((e={})[k]=x,e.centerOffset=x-O,e)}},effect:function(t){var e=t.state,i=t.options.element,n=void 0===i?"[data-popper-arrow]":i;null!=n&&("string"!=typeof n||(n=e.elements.popper.querySelector(n)))&&W(e.elements.popper,n)&&(e.elements.arrow=n)},requires:["popperOffsets"],requiresIfExists:["preventOverflow"]};function Z(t){return t.split("-")[1]}var tt={top:"auto",right:"auto",bottom:"auto",left:"auto"};function et(t){var e,i=t.popper,a=t.popperRect,l=t.placement,c=t.variation,d=t.offsets,u=t.position,f=t.gpuAcceleration,p=t.adaptive,m=t.roundOffsets,g=t.isFixed,_=d.x,b=void 0===_?0:_,v=d.y,y=void 0===v?0:v,w="function"==typeof m?m({x:b,y}):{x:b,y};b=w.x,y=w.y;var E=d.hasOwnProperty("x"),A=d.hasOwnProperty("y"),T=r,C=n,O=window;if(p){var x=K(i),L="clientHeight",S="clientWidth";x===k(i)&&"static"!==z(x=q(i)).position&&"absolute"===u&&(L="scrollHeight",S="scrollWidth"),(l===n||(l===r||l===o)&&c===h)&&(C=s,y-=(g&&x===O&&O.visualViewport?O.visualViewport.height:x[L])-a.height,y*=f?1:-1),l!==r&&(l!==n&&l!==s||c!==h)||(T=o,b-=(g&&x===O&&O.visualViewport?O.visualViewport.width:x[S])-a.width,b*=f?1:-1)}var D,$=Object.assign({position:u},p&&tt),I=!0===m?function(t,e){var i=t.x,n=t.y,s=e.devicePixelRatio||1;return{x:M(i*s)/s||0,y:M(n*s)/s||0}}({x:b,y},k(i)):{x:b,y};return b=I.x,y=I.y,f?Object.assign({},$,((D={})[C]=A?"0":"",D[T]=E?"0":"",D.transform=(O.devicePixelRatio||1)<=1?"translate("+b+"px, "+y+"px)":"translate3d("+b+"px, "+y+"px, 0)",D)):Object.assign({},$,((e={})[C]=A?y+"px":"",e[T]=E?b+"px":"",e.transform="",e))}const it={name:"computeStyles",enabled:!0,phase:"beforeWrite",fn:function(t){var e=t.state,i=t.options,n=i.gpuAcceleration,s=void 0===n||n,o=i.adaptive,r=void 0===o||o,a=i.roundOffsets,l=void 0===a||a,c={placement:I(e.placement),variation:Z(e.placement),popper:e.elements.popper,popperRect:e.rects.popper,gpuAcceleration:s,isFixed:"fixed"===e.options.strategy};null!=e.modifiersData.popperOffsets&&(e.styles.popper=Object.assign({},e.styles.popper,et(Object.assign({},c,{offsets:e.modifiersData.popperOffsets,position:e.options.strategy,adaptive:r,roundOffsets:l})))),null!=e.modifiersData.arrow&&(e.styles.arrow=Object.assign({},e.styles.arrow,et(Object.assign({},c,{offsets:e.modifiersData.arrow,position:"absolute",adaptive:!1,roundOffsets:l})))),e.attributes.popper=Object.assign({},e.attributes.popper,{"data-popper-placement":e.placement})},data:{}};var nt={passive:!0};const st={name:"eventListeners",enabled:!0,phase:"write",fn:function(){},effect:function(t){var e=t.state,i=t.instance,n=t.options,s=n.scroll,o=void 0===s||s,r=n.resize,a=void 0===r||r,l=k(e.elements.popper),c=[].concat(e.scrollParents.reference,e.scrollParents.popper);return o&&c.forEach((function(t){t.addEventListener("scroll",i.update,nt)})),a&&l.addEventListener("resize",i.update,nt),function(){o&&c.forEach((function(t){t.removeEventListener("scroll",i.update,nt)})),a&&l.removeEventListener("resize",i.update,nt)}},data:{}};var ot={left:"right",right:"left",bottom:"top",top:"bottom"};function rt(t){return t.replace(/left|right|bottom|top/g,(function(t){return ot[t]}))}var at={start:"end",end:"start"};function lt(t){return t.replace(/start|end/g,(function(t){return at[t]}))}function ct(t){var e=k(t);return{scrollLeft:e.pageXOffset,scrollTop:e.pageYOffset}}function ht(t){return H(q(t)).left+ct(t).scrollLeft}function dt(t){var e=z(t),i=e.overflow,n=e.overflowX,s=e.overflowY;return/auto|scroll|overlay|hidden/.test(i+s+n)}function ut(t){return["html","body","#document"].indexOf(x(t))>=0?t.ownerDocument.body:S(t)&&dt(t)?t:ut(V(t))}function ft(t,e){var i;void 0===e&&(e=[]);var n=ut(t),s=n===(null==(i=t.ownerDocument)?void 0:i.body),o=k(n),r=s?[o].concat(o.visualViewport||[],dt(n)?n:[]):n,a=e.concat(r);return s?a:a.concat(ft(V(r)))}function pt(t){return Object.assign({},t,{left:t.x,top:t.y,right:t.x+t.width,bottom:t.y+t.height})}function mt(t,e,i){return e===u?pt(function(t,e){var i=k(t),n=q(t),s=i.visualViewport,o=n.clientWidth,r=n.clientHeight,a=0,l=0;if(s){o=s.width,r=s.height;var c=F();(c||!c&&"fixed"===e)&&(a=s.offsetLeft,l=s.offsetTop)}return{width:o,height:r,x:a+ht(t),y:l}}(t,i)):L(e)?function(t,e){var i=H(t,!1,"fixed"===e);return i.top=i.top+t.clientTop,i.left=i.left+t.clientLeft,i.bottom=i.top+t.clientHeight,i.right=i.left+t.clientWidth,i.width=t.clientWidth,i.height=t.clientHeight,i.x=i.left,i.y=i.top,i}(e,i):pt(function(t){var e,i=q(t),n=ct(t),s=null==(e=t.ownerDocument)?void 0:e.body,o=N(i.scrollWidth,i.clientWidth,s?s.scrollWidth:0,s?s.clientWidth:0),r=N(i.scrollHeight,i.clientHeight,s?s.scrollHeight:0,s?s.clientHeight:0),a=-n.scrollLeft+ht(t),l=-n.scrollTop;return"rtl"===z(s||i).direction&&(a+=N(i.clientWidth,s?s.clientWidth:0)-o),{width:o,height:r,x:a,y:l}}(q(t)))}function gt(t){var e,i=t.reference,a=t.element,l=t.placement,d=l?I(l):null,u=l?Z(l):null,f=i.x+i.width/2-a.width/2,p=i.y+i.height/2-a.height/2;switch(d){case n:e={x:f,y:i.y-a.height};break;case s:e={x:f,y:i.y+i.height};break;case o:e={x:i.x+i.width,y:p};break;case r:e={x:i.x-a.width,y:p};break;default:e={x:i.x,y:i.y}}var m=d?Q(d):null;if(null!=m){var g="y"===m?"height":"width";switch(u){case c:e[m]=e[m]-(i[g]/2-a[g]/2);break;case h:e[m]=e[m]+(i[g]/2-a[g]/2)}}return e}function _t(t,e){void 0===e&&(e={});var i=e,r=i.placement,a=void 0===r?t.placement:r,c=i.strategy,h=void 0===c?t.strategy:c,m=i.boundary,g=void 0===m?d:m,_=i.rootBoundary,b=void 0===_?u:_,v=i.elementContext,y=void 0===v?f:v,w=i.altBoundary,E=void 0!==w&&w,A=i.padding,T=void 0===A?0:A,C=U("number"!=typeof T?T:G(T,l)),O=y===f?p:f,k=t.rects.popper,D=t.elements[E?O:y],$=function(t,e,i,n){var s="clippingParents"===e?function(t){var e=ft(V(t)),i=["absolute","fixed"].indexOf(z(t).position)>=0&&S(t)?K(t):t;return L(i)?e.filter((function(t){return L(t)&&W(t,i)&&"body"!==x(t)})):[]}(t):[].concat(e),o=[].concat(s,[i]),r=o[0],a=o.reduce((function(e,i){var s=mt(t,i,n);return e.top=N(s.top,e.top),e.right=P(s.right,e.right),e.bottom=P(s.bottom,e.bottom),e.left=N(s.left,e.left),e}),mt(t,r,n));return a.width=a.right-a.left,a.height=a.bottom-a.top,a.x=a.left,a.y=a.top,a}(L(D)?D:D.contextElement||q(t.elements.popper),g,b,h),I=H(t.elements.reference),M=gt({reference:I,element:k,strategy:"absolute",placement:a}),j=pt(Object.assign({},k,M)),F=y===f?j:I,B={top:$.top-F.top+C.top,bottom:F.bottom-$.bottom+C.bottom,left:$.left-F.left+C.left,right:F.right-$.right+C.right},R=t.modifiersData.offset;if(y===f&&R){var Y=R[a];Object.keys(B).forEach((function(t){var e=[o,s].indexOf(t)>=0?1:-1,i=[n,s].indexOf(t)>=0?"y":"x";B[t]+=Y[i]*e}))}return B}const bt={name:"flip",enabled:!0,phase:"main",fn:function(t){var e=t.state,i=t.options,h=t.name;if(!e.modifiersData[h]._skip){for(var d=i.mainAxis,u=void 0===d||d,f=i.altAxis,p=void 0===f||f,_=i.fallbackPlacements,b=i.padding,v=i.boundary,y=i.rootBoundary,w=i.altBoundary,E=i.flipVariations,A=void 0===E||E,T=i.allowedAutoPlacements,C=e.options.placement,O=I(C),x=_||(O!==C&&A?function(t){if(I(t)===a)return[];var e=rt(t);return[lt(t),e,lt(e)]}(C):[rt(C)]),k=[C].concat(x).reduce((function(t,i){return t.concat(I(i)===a?function(t,e){void 0===e&&(e={});var i=e,n=i.placement,s=i.boundary,o=i.rootBoundary,r=i.padding,a=i.flipVariations,c=i.allowedAutoPlacements,h=void 0===c?g:c,d=Z(n),u=d?a?m:m.filter((function(t){return Z(t)===d})):l,f=u.filter((function(t){return h.indexOf(t)>=0}));0===f.length&&(f=u);var p=f.reduce((function(e,i){return e[i]=_t(t,{placement:i,boundary:s,rootBoundary:o,padding:r})[I(i)],e}),{});return Object.keys(p).sort((function(t,e){return p[t]-p[e]}))}(e,{placement:i,boundary:v,rootBoundary:y,padding:b,flipVariations:A,allowedAutoPlacements:T}):i)}),[]),L=e.rects.reference,S=e.rects.popper,D=new Map,$=!0,N=k[0],P=0;P=0,B=H?"width":"height",W=_t(e,{placement:M,boundary:v,rootBoundary:y,altBoundary:w,padding:b}),z=H?F?o:r:F?s:n;L[B]>S[B]&&(z=rt(z));var R=rt(z),q=[];if(u&&q.push(W[j]<=0),p&&q.push(W[z]<=0,W[R]<=0),q.every((function(t){return t}))){N=M,$=!1;break}D.set(M,q)}if($)for(var V=function(t){var e=k.find((function(e){var i=D.get(e);if(i)return i.slice(0,t).every((function(t){return t}))}));if(e)return N=e,"break"},Y=A?3:1;Y>0&&"break"!==V(Y);Y--);e.placement!==N&&(e.modifiersData[h]._skip=!0,e.placement=N,e.reset=!0)}},requiresIfExists:["offset"],data:{_skip:!1}};function vt(t,e,i){return void 0===i&&(i={x:0,y:0}),{top:t.top-e.height-i.y,right:t.right-e.width+i.x,bottom:t.bottom-e.height+i.y,left:t.left-e.width-i.x}}function yt(t){return[n,o,s,r].some((function(e){return t[e]>=0}))}const wt={name:"hide",enabled:!0,phase:"main",requiresIfExists:["preventOverflow"],fn:function(t){var e=t.state,i=t.name,n=e.rects.reference,s=e.rects.popper,o=e.modifiersData.preventOverflow,r=_t(e,{elementContext:"reference"}),a=_t(e,{altBoundary:!0}),l=vt(r,n),c=vt(a,s,o),h=yt(l),d=yt(c);e.modifiersData[i]={referenceClippingOffsets:l,popperEscapeOffsets:c,isReferenceHidden:h,hasPopperEscaped:d},e.attributes.popper=Object.assign({},e.attributes.popper,{"data-popper-reference-hidden":h,"data-popper-escaped":d})}},Et={name:"offset",enabled:!0,phase:"main",requires:["popperOffsets"],fn:function(t){var e=t.state,i=t.options,s=t.name,a=i.offset,l=void 0===a?[0,0]:a,c=g.reduce((function(t,i){return t[i]=function(t,e,i){var s=I(t),a=[r,n].indexOf(s)>=0?-1:1,l="function"==typeof i?i(Object.assign({},e,{placement:t})):i,c=l[0],h=l[1];return c=c||0,h=(h||0)*a,[r,o].indexOf(s)>=0?{x:h,y:c}:{x:c,y:h}}(i,e.rects,l),t}),{}),h=c[e.placement],d=h.x,u=h.y;null!=e.modifiersData.popperOffsets&&(e.modifiersData.popperOffsets.x+=d,e.modifiersData.popperOffsets.y+=u),e.modifiersData[s]=c}},At={name:"popperOffsets",enabled:!0,phase:"read",fn:function(t){var e=t.state,i=t.name;e.modifiersData[i]=gt({reference:e.rects.reference,element:e.rects.popper,strategy:"absolute",placement:e.placement})},data:{}},Tt={name:"preventOverflow",enabled:!0,phase:"main",fn:function(t){var e=t.state,i=t.options,a=t.name,l=i.mainAxis,h=void 0===l||l,d=i.altAxis,u=void 0!==d&&d,f=i.boundary,p=i.rootBoundary,m=i.altBoundary,g=i.padding,_=i.tether,b=void 0===_||_,v=i.tetherOffset,y=void 0===v?0:v,w=_t(e,{boundary:f,rootBoundary:p,padding:g,altBoundary:m}),E=I(e.placement),A=Z(e.placement),T=!A,C=Q(E),O="x"===C?"y":"x",x=e.modifiersData.popperOffsets,k=e.rects.reference,L=e.rects.popper,S="function"==typeof y?y(Object.assign({},e.rects,{placement:e.placement})):y,D="number"==typeof S?{mainAxis:S,altAxis:S}:Object.assign({mainAxis:0,altAxis:0},S),$=e.modifiersData.offset?e.modifiersData.offset[e.placement]:null,M={x:0,y:0};if(x){if(h){var j,F="y"===C?n:r,H="y"===C?s:o,W="y"===C?"height":"width",z=x[C],R=z+w[F],q=z-w[H],V=b?-L[W]/2:0,Y=A===c?k[W]:L[W],U=A===c?-L[W]:-k[W],G=e.elements.arrow,J=b&&G?B(G):{width:0,height:0},tt=e.modifiersData["arrow#persistent"]?e.modifiersData["arrow#persistent"].padding:{top:0,right:0,bottom:0,left:0},et=tt[F],it=tt[H],nt=X(0,k[W],J[W]),st=T?k[W]/2-V-nt-et-D.mainAxis:Y-nt-et-D.mainAxis,ot=T?-k[W]/2+V+nt+it+D.mainAxis:U+nt+it+D.mainAxis,rt=e.elements.arrow&&K(e.elements.arrow),at=rt?"y"===C?rt.clientTop||0:rt.clientLeft||0:0,lt=null!=(j=null==$?void 0:$[C])?j:0,ct=z+ot-lt,ht=X(b?P(R,z+st-lt-at):R,z,b?N(q,ct):q);x[C]=ht,M[C]=ht-z}if(u){var dt,ut="x"===C?n:r,ft="x"===C?s:o,pt=x[O],mt="y"===O?"height":"width",gt=pt+w[ut],bt=pt-w[ft],vt=-1!==[n,r].indexOf(E),yt=null!=(dt=null==$?void 0:$[O])?dt:0,wt=vt?gt:pt-k[mt]-L[mt]-yt+D.altAxis,Et=vt?pt+k[mt]+L[mt]-yt-D.altAxis:bt,At=b&&vt?function(t,e,i){var n=X(t,e,i);return n>i?i:n}(wt,pt,Et):X(b?wt:gt,pt,b?Et:bt);x[O]=At,M[O]=At-pt}e.modifiersData[a]=M}},requiresIfExists:["offset"]};function Ct(t,e,i){void 0===i&&(i=!1);var n,s,o=S(e),r=S(e)&&function(t){var e=t.getBoundingClientRect(),i=M(e.width)/t.offsetWidth||1,n=M(e.height)/t.offsetHeight||1;return 1!==i||1!==n}(e),a=q(e),l=H(t,r,i),c={scrollLeft:0,scrollTop:0},h={x:0,y:0};return(o||!o&&!i)&&(("body"!==x(e)||dt(a))&&(c=(n=e)!==k(n)&&S(n)?{scrollLeft:(s=n).scrollLeft,scrollTop:s.scrollTop}:ct(n)),S(e)?((h=H(e,!0)).x+=e.clientLeft,h.y+=e.clientTop):a&&(h.x=ht(a))),{x:l.left+c.scrollLeft-h.x,y:l.top+c.scrollTop-h.y,width:l.width,height:l.height}}function Ot(t){var e=new Map,i=new Set,n=[];function s(t){i.add(t.name),[].concat(t.requires||[],t.requiresIfExists||[]).forEach((function(t){if(!i.has(t)){var n=e.get(t);n&&s(n)}})),n.push(t)}return t.forEach((function(t){e.set(t.name,t)})),t.forEach((function(t){i.has(t.name)||s(t)})),n}var xt={placement:"bottom",modifiers:[],strategy:"absolute"};function kt(){for(var t=arguments.length,e=new Array(t),i=0;iIt.has(t)&&It.get(t).get(e)||null,remove(t,e){if(!It.has(t))return;const i=It.get(t);i.delete(e),0===i.size&&It.delete(t)}},Pt="transitionend",Mt=t=>(t&&window.CSS&&window.CSS.escape&&(t=t.replace(/#([^\s"#']+)/g,((t,e)=>`#${CSS.escape(e)}`))),t),jt=t=>{t.dispatchEvent(new Event(Pt))},Ft=t=>!(!t||"object"!=typeof t)&&(void 0!==t.jquery&&(t=t[0]),void 0!==t.nodeType),Ht=t=>Ft(t)?t.jquery?t[0]:t:"string"==typeof t&&t.length>0?document.querySelector(Mt(t)):null,Bt=t=>{if(!Ft(t)||0===t.getClientRects().length)return!1;const e="visible"===getComputedStyle(t).getPropertyValue("visibility"),i=t.closest("details:not([open])");if(!i)return e;if(i!==t){const e=t.closest("summary");if(e&&e.parentNode!==i)return!1;if(null===e)return!1}return e},Wt=t=>!t||t.nodeType!==Node.ELEMENT_NODE||!!t.classList.contains("disabled")||(void 0!==t.disabled?t.disabled:t.hasAttribute("disabled")&&"false"!==t.getAttribute("disabled")),zt=t=>{if(!document.documentElement.attachShadow)return null;if("function"==typeof t.getRootNode){const e=t.getRootNode();return e instanceof ShadowRoot?e:null}return t instanceof ShadowRoot?t:t.parentNode?zt(t.parentNode):null},Rt=()=>{},qt=t=>{t.offsetHeight},Vt=()=>window.jQuery&&!document.body.hasAttribute("data-bs-no-jquery")?window.jQuery:null,Yt=[],Kt=()=>"rtl"===document.documentElement.dir,Qt=t=>{var e;e=()=>{const e=Vt();if(e){const i=t.NAME,n=e.fn[i];e.fn[i]=t.jQueryInterface,e.fn[i].Constructor=t,e.fn[i].noConflict=()=>(e.fn[i]=n,t.jQueryInterface)}},"loading"===document.readyState?(Yt.length||document.addEventListener("DOMContentLoaded",(()=>{for(const t of Yt)t()})),Yt.push(e)):e()},Xt=(t,e=[],i=t)=>"function"==typeof t?t(...e):i,Ut=(t,e,i=!0)=>{if(!i)return void Xt(t);const n=(t=>{if(!t)return 0;let{transitionDuration:e,transitionDelay:i}=window.getComputedStyle(t);const n=Number.parseFloat(e),s=Number.parseFloat(i);return n||s?(e=e.split(",")[0],i=i.split(",")[0],1e3*(Number.parseFloat(e)+Number.parseFloat(i))):0})(e)+5;let s=!1;const o=({target:i})=>{i===e&&(s=!0,e.removeEventListener(Pt,o),Xt(t))};e.addEventListener(Pt,o),setTimeout((()=>{s||jt(e)}),n)},Gt=(t,e,i,n)=>{const s=t.length;let o=t.indexOf(e);return-1===o?!i&&n?t[s-1]:t[0]:(o+=i?1:-1,n&&(o=(o+s)%s),t[Math.max(0,Math.min(o,s-1))])},Jt=/[^.]*(?=\..*)\.|.*/,Zt=/\..*/,te=/::\d+$/,ee={};let ie=1;const ne={mouseenter:"mouseover",mouseleave:"mouseout"},se=new Set(["click","dblclick","mouseup","mousedown","contextmenu","mousewheel","DOMMouseScroll","mouseover","mouseout","mousemove","selectstart","selectend","keydown","keypress","keyup","orientationchange","touchstart","touchmove","touchend","touchcancel","pointerdown","pointermove","pointerup","pointerleave","pointercancel","gesturestart","gesturechange","gestureend","focus","blur","change","reset","select","submit","focusin","focusout","load","unload","beforeunload","resize","move","DOMContentLoaded","readystatechange","error","abort","scroll"]);function oe(t,e){return e&&`${e}::${ie++}`||t.uidEvent||ie++}function re(t){const e=oe(t);return t.uidEvent=e,ee[e]=ee[e]||{},ee[e]}function ae(t,e,i=null){return Object.values(t).find((t=>t.callable===e&&t.delegationSelector===i))}function le(t,e,i){const n="string"==typeof e,s=n?i:e||i;let o=ue(t);return se.has(o)||(o=t),[n,s,o]}function ce(t,e,i,n,s){if("string"!=typeof e||!t)return;let[o,r,a]=le(e,i,n);if(e in ne){const t=t=>function(e){if(!e.relatedTarget||e.relatedTarget!==e.delegateTarget&&!e.delegateTarget.contains(e.relatedTarget))return t.call(this,e)};r=t(r)}const l=re(t),c=l[a]||(l[a]={}),h=ae(c,r,o?i:null);if(h)return void(h.oneOff=h.oneOff&&s);const d=oe(r,e.replace(Jt,"")),u=o?function(t,e,i){return function n(s){const o=t.querySelectorAll(e);for(let{target:r}=s;r&&r!==this;r=r.parentNode)for(const a of o)if(a===r)return pe(s,{delegateTarget:r}),n.oneOff&&fe.off(t,s.type,e,i),i.apply(r,[s])}}(t,i,r):function(t,e){return function i(n){return pe(n,{delegateTarget:t}),i.oneOff&&fe.off(t,n.type,e),e.apply(t,[n])}}(t,r);u.delegationSelector=o?i:null,u.callable=r,u.oneOff=s,u.uidEvent=d,c[d]=u,t.addEventListener(a,u,o)}function he(t,e,i,n,s){const o=ae(e[i],n,s);o&&(t.removeEventListener(i,o,Boolean(s)),delete e[i][o.uidEvent])}function de(t,e,i,n){const s=e[i]||{};for(const[o,r]of Object.entries(s))o.includes(n)&&he(t,e,i,r.callable,r.delegationSelector)}function ue(t){return t=t.replace(Zt,""),ne[t]||t}const fe={on(t,e,i,n){ce(t,e,i,n,!1)},one(t,e,i,n){ce(t,e,i,n,!0)},off(t,e,i,n){if("string"!=typeof e||!t)return;const[s,o,r]=le(e,i,n),a=r!==e,l=re(t),c=l[r]||{},h=e.startsWith(".");if(void 0===o){if(h)for(const i of Object.keys(l))de(t,l,i,e.slice(1));for(const[i,n]of Object.entries(c)){const s=i.replace(te,"");a&&!e.includes(s)||he(t,l,r,n.callable,n.delegationSelector)}}else{if(!Object.keys(c).length)return;he(t,l,r,o,s?i:null)}},trigger(t,e,i){if("string"!=typeof e||!t)return null;const n=Vt();let s=null,o=!0,r=!0,a=!1;e!==ue(e)&&n&&(s=n.Event(e,i),n(t).trigger(s),o=!s.isPropagationStopped(),r=!s.isImmediatePropagationStopped(),a=s.isDefaultPrevented());const l=pe(new Event(e,{bubbles:o,cancelable:!0}),i);return a&&l.preventDefault(),r&&t.dispatchEvent(l),l.defaultPrevented&&s&&s.preventDefault(),l}};function pe(t,e={}){for(const[i,n]of Object.entries(e))try{t[i]=n}catch(e){Object.defineProperty(t,i,{configurable:!0,get:()=>n})}return t}function me(t){if("true"===t)return!0;if("false"===t)return!1;if(t===Number(t).toString())return Number(t);if(""===t||"null"===t)return null;if("string"!=typeof t)return t;try{return JSON.parse(decodeURIComponent(t))}catch(e){return t}}function ge(t){return t.replace(/[A-Z]/g,(t=>`-${t.toLowerCase()}`))}const _e={setDataAttribute(t,e,i){t.setAttribute(`data-bs-${ge(e)}`,i)},removeDataAttribute(t,e){t.removeAttribute(`data-bs-${ge(e)}`)},getDataAttributes(t){if(!t)return{};const e={},i=Object.keys(t.dataset).filter((t=>t.startsWith("bs")&&!t.startsWith("bsConfig")));for(const n of i){let i=n.replace(/^bs/,"");i=i.charAt(0).toLowerCase()+i.slice(1,i.length),e[i]=me(t.dataset[n])}return e},getDataAttribute:(t,e)=>me(t.getAttribute(`data-bs-${ge(e)}`))};class be{static get Default(){return{}}static get DefaultType(){return{}}static get NAME(){throw new Error('You have to implement the static method "NAME", for each component!')}_getConfig(t){return t=this._mergeConfigObj(t),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}_configAfterMerge(t){return t}_mergeConfigObj(t,e){const i=Ft(e)?_e.getDataAttribute(e,"config"):{};return{...this.constructor.Default,..."object"==typeof i?i:{},...Ft(e)?_e.getDataAttributes(e):{},..."object"==typeof t?t:{}}}_typeCheckConfig(t,e=this.constructor.DefaultType){for(const[n,s]of Object.entries(e)){const e=t[n],o=Ft(e)?"element":null==(i=e)?`${i}`:Object.prototype.toString.call(i).match(/\s([a-z]+)/i)[1].toLowerCase();if(!new RegExp(s).test(o))throw new TypeError(`${this.constructor.NAME.toUpperCase()}: Option "${n}" provided type "${o}" but expected type "${s}".`)}var i}}class ve extends be{constructor(t,e){super(),(t=Ht(t))&&(this._element=t,this._config=this._getConfig(e),Nt.set(this._element,this.constructor.DATA_KEY,this))}dispose(){Nt.remove(this._element,this.constructor.DATA_KEY),fe.off(this._element,this.constructor.EVENT_KEY);for(const t of Object.getOwnPropertyNames(this))this[t]=null}_queueCallback(t,e,i=!0){Ut(t,e,i)}_getConfig(t){return t=this._mergeConfigObj(t,this._element),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}static getInstance(t){return Nt.get(Ht(t),this.DATA_KEY)}static getOrCreateInstance(t,e={}){return this.getInstance(t)||new this(t,"object"==typeof e?e:null)}static get VERSION(){return"5.3.3"}static get DATA_KEY(){return`bs.${this.NAME}`}static get EVENT_KEY(){return`.${this.DATA_KEY}`}static eventName(t){return`${t}${this.EVENT_KEY}`}}const ye=t=>{let e=t.getAttribute("data-bs-target");if(!e||"#"===e){let i=t.getAttribute("href");if(!i||!i.includes("#")&&!i.startsWith("."))return null;i.includes("#")&&!i.startsWith("#")&&(i=`#${i.split("#")[1]}`),e=i&&"#"!==i?i.trim():null}return e?e.split(",").map((t=>Mt(t))).join(","):null},we={find:(t,e=document.documentElement)=>[].concat(...Element.prototype.querySelectorAll.call(e,t)),findOne:(t,e=document.documentElement)=>Element.prototype.querySelector.call(e,t),children:(t,e)=>[].concat(...t.children).filter((t=>t.matches(e))),parents(t,e){const i=[];let n=t.parentNode.closest(e);for(;n;)i.push(n),n=n.parentNode.closest(e);return i},prev(t,e){let i=t.previousElementSibling;for(;i;){if(i.matches(e))return[i];i=i.previousElementSibling}return[]},next(t,e){let i=t.nextElementSibling;for(;i;){if(i.matches(e))return[i];i=i.nextElementSibling}return[]},focusableChildren(t){const e=["a","button","input","textarea","select","details","[tabindex]",'[contenteditable="true"]'].map((t=>`${t}:not([tabindex^="-"])`)).join(",");return this.find(e,t).filter((t=>!Wt(t)&&Bt(t)))},getSelectorFromElement(t){const e=ye(t);return e&&we.findOne(e)?e:null},getElementFromSelector(t){const e=ye(t);return e?we.findOne(e):null},getMultipleElementsFromSelector(t){const e=ye(t);return e?we.find(e):[]}},Ee=(t,e="hide")=>{const i=`click.dismiss${t.EVENT_KEY}`,n=t.NAME;fe.on(document,i,`[data-bs-dismiss="${n}"]`,(function(i){if(["A","AREA"].includes(this.tagName)&&i.preventDefault(),Wt(this))return;const s=we.getElementFromSelector(this)||this.closest(`.${n}`);t.getOrCreateInstance(s)[e]()}))},Ae=".bs.alert",Te=`close${Ae}`,Ce=`closed${Ae}`;class Oe extends ve{static get NAME(){return"alert"}close(){if(fe.trigger(this._element,Te).defaultPrevented)return;this._element.classList.remove("show");const t=this._element.classList.contains("fade");this._queueCallback((()=>this._destroyElement()),this._element,t)}_destroyElement(){this._element.remove(),fe.trigger(this._element,Ce),this.dispose()}static jQueryInterface(t){return this.each((function(){const e=Oe.getOrCreateInstance(this);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}Ee(Oe,"close"),Qt(Oe);const xe='[data-bs-toggle="button"]';class ke extends ve{static get NAME(){return"button"}toggle(){this._element.setAttribute("aria-pressed",this._element.classList.toggle("active"))}static jQueryInterface(t){return this.each((function(){const e=ke.getOrCreateInstance(this);"toggle"===t&&e[t]()}))}}fe.on(document,"click.bs.button.data-api",xe,(t=>{t.preventDefault();const e=t.target.closest(xe);ke.getOrCreateInstance(e).toggle()})),Qt(ke);const Le=".bs.swipe",Se=`touchstart${Le}`,De=`touchmove${Le}`,$e=`touchend${Le}`,Ie=`pointerdown${Le}`,Ne=`pointerup${Le}`,Pe={endCallback:null,leftCallback:null,rightCallback:null},Me={endCallback:"(function|null)",leftCallback:"(function|null)",rightCallback:"(function|null)"};class je extends be{constructor(t,e){super(),this._element=t,t&&je.isSupported()&&(this._config=this._getConfig(e),this._deltaX=0,this._supportPointerEvents=Boolean(window.PointerEvent),this._initEvents())}static get Default(){return Pe}static get DefaultType(){return Me}static get NAME(){return"swipe"}dispose(){fe.off(this._element,Le)}_start(t){this._supportPointerEvents?this._eventIsPointerPenTouch(t)&&(this._deltaX=t.clientX):this._deltaX=t.touches[0].clientX}_end(t){this._eventIsPointerPenTouch(t)&&(this._deltaX=t.clientX-this._deltaX),this._handleSwipe(),Xt(this._config.endCallback)}_move(t){this._deltaX=t.touches&&t.touches.length>1?0:t.touches[0].clientX-this._deltaX}_handleSwipe(){const t=Math.abs(this._deltaX);if(t<=40)return;const e=t/this._deltaX;this._deltaX=0,e&&Xt(e>0?this._config.rightCallback:this._config.leftCallback)}_initEvents(){this._supportPointerEvents?(fe.on(this._element,Ie,(t=>this._start(t))),fe.on(this._element,Ne,(t=>this._end(t))),this._element.classList.add("pointer-event")):(fe.on(this._element,Se,(t=>this._start(t))),fe.on(this._element,De,(t=>this._move(t))),fe.on(this._element,$e,(t=>this._end(t))))}_eventIsPointerPenTouch(t){return this._supportPointerEvents&&("pen"===t.pointerType||"touch"===t.pointerType)}static isSupported(){return"ontouchstart"in document.documentElement||navigator.maxTouchPoints>0}}const Fe=".bs.carousel",He=".data-api",Be="ArrowLeft",We="ArrowRight",ze="next",Re="prev",qe="left",Ve="right",Ye=`slide${Fe}`,Ke=`slid${Fe}`,Qe=`keydown${Fe}`,Xe=`mouseenter${Fe}`,Ue=`mouseleave${Fe}`,Ge=`dragstart${Fe}`,Je=`load${Fe}${He}`,Ze=`click${Fe}${He}`,ti="carousel",ei="active",ii=".active",ni=".carousel-item",si=ii+ni,oi={[Be]:Ve,[We]:qe},ri={interval:5e3,keyboard:!0,pause:"hover",ride:!1,touch:!0,wrap:!0},ai={interval:"(number|boolean)",keyboard:"boolean",pause:"(string|boolean)",ride:"(boolean|string)",touch:"boolean",wrap:"boolean"};class li extends ve{constructor(t,e){super(t,e),this._interval=null,this._activeElement=null,this._isSliding=!1,this.touchTimeout=null,this._swipeHelper=null,this._indicatorsElement=we.findOne(".carousel-indicators",this._element),this._addEventListeners(),this._config.ride===ti&&this.cycle()}static get Default(){return ri}static get DefaultType(){return ai}static get NAME(){return"carousel"}next(){this._slide(ze)}nextWhenVisible(){!document.hidden&&Bt(this._element)&&this.next()}prev(){this._slide(Re)}pause(){this._isSliding&&jt(this._element),this._clearInterval()}cycle(){this._clearInterval(),this._updateInterval(),this._interval=setInterval((()=>this.nextWhenVisible()),this._config.interval)}_maybeEnableCycle(){this._config.ride&&(this._isSliding?fe.one(this._element,Ke,(()=>this.cycle())):this.cycle())}to(t){const e=this._getItems();if(t>e.length-1||t<0)return;if(this._isSliding)return void fe.one(this._element,Ke,(()=>this.to(t)));const i=this._getItemIndex(this._getActive());if(i===t)return;const n=t>i?ze:Re;this._slide(n,e[t])}dispose(){this._swipeHelper&&this._swipeHelper.dispose(),super.dispose()}_configAfterMerge(t){return t.defaultInterval=t.interval,t}_addEventListeners(){this._config.keyboard&&fe.on(this._element,Qe,(t=>this._keydown(t))),"hover"===this._config.pause&&(fe.on(this._element,Xe,(()=>this.pause())),fe.on(this._element,Ue,(()=>this._maybeEnableCycle()))),this._config.touch&&je.isSupported()&&this._addTouchEventListeners()}_addTouchEventListeners(){for(const t of we.find(".carousel-item img",this._element))fe.on(t,Ge,(t=>t.preventDefault()));const t={leftCallback:()=>this._slide(this._directionToOrder(qe)),rightCallback:()=>this._slide(this._directionToOrder(Ve)),endCallback:()=>{"hover"===this._config.pause&&(this.pause(),this.touchTimeout&&clearTimeout(this.touchTimeout),this.touchTimeout=setTimeout((()=>this._maybeEnableCycle()),500+this._config.interval))}};this._swipeHelper=new je(this._element,t)}_keydown(t){if(/input|textarea/i.test(t.target.tagName))return;const e=oi[t.key];e&&(t.preventDefault(),this._slide(this._directionToOrder(e)))}_getItemIndex(t){return this._getItems().indexOf(t)}_setActiveIndicatorElement(t){if(!this._indicatorsElement)return;const e=we.findOne(ii,this._indicatorsElement);e.classList.remove(ei),e.removeAttribute("aria-current");const i=we.findOne(`[data-bs-slide-to="${t}"]`,this._indicatorsElement);i&&(i.classList.add(ei),i.setAttribute("aria-current","true"))}_updateInterval(){const t=this._activeElement||this._getActive();if(!t)return;const e=Number.parseInt(t.getAttribute("data-bs-interval"),10);this._config.interval=e||this._config.defaultInterval}_slide(t,e=null){if(this._isSliding)return;const i=this._getActive(),n=t===ze,s=e||Gt(this._getItems(),i,n,this._config.wrap);if(s===i)return;const o=this._getItemIndex(s),r=e=>fe.trigger(this._element,e,{relatedTarget:s,direction:this._orderToDirection(t),from:this._getItemIndex(i),to:o});if(r(Ye).defaultPrevented)return;if(!i||!s)return;const a=Boolean(this._interval);this.pause(),this._isSliding=!0,this._setActiveIndicatorElement(o),this._activeElement=s;const l=n?"carousel-item-start":"carousel-item-end",c=n?"carousel-item-next":"carousel-item-prev";s.classList.add(c),qt(s),i.classList.add(l),s.classList.add(l),this._queueCallback((()=>{s.classList.remove(l,c),s.classList.add(ei),i.classList.remove(ei,c,l),this._isSliding=!1,r(Ke)}),i,this._isAnimated()),a&&this.cycle()}_isAnimated(){return this._element.classList.contains("slide")}_getActive(){return we.findOne(si,this._element)}_getItems(){return we.find(ni,this._element)}_clearInterval(){this._interval&&(clearInterval(this._interval),this._interval=null)}_directionToOrder(t){return Kt()?t===qe?Re:ze:t===qe?ze:Re}_orderToDirection(t){return Kt()?t===Re?qe:Ve:t===Re?Ve:qe}static jQueryInterface(t){return this.each((function(){const e=li.getOrCreateInstance(this,t);if("number"!=typeof t){if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}else e.to(t)}))}}fe.on(document,Ze,"[data-bs-slide], [data-bs-slide-to]",(function(t){const e=we.getElementFromSelector(this);if(!e||!e.classList.contains(ti))return;t.preventDefault();const i=li.getOrCreateInstance(e),n=this.getAttribute("data-bs-slide-to");return n?(i.to(n),void i._maybeEnableCycle()):"next"===_e.getDataAttribute(this,"slide")?(i.next(),void i._maybeEnableCycle()):(i.prev(),void i._maybeEnableCycle())})),fe.on(window,Je,(()=>{const t=we.find('[data-bs-ride="carousel"]');for(const e of t)li.getOrCreateInstance(e)})),Qt(li);const ci=".bs.collapse",hi=`show${ci}`,di=`shown${ci}`,ui=`hide${ci}`,fi=`hidden${ci}`,pi=`click${ci}.data-api`,mi="show",gi="collapse",_i="collapsing",bi=`:scope .${gi} .${gi}`,vi='[data-bs-toggle="collapse"]',yi={parent:null,toggle:!0},wi={parent:"(null|element)",toggle:"boolean"};class Ei extends ve{constructor(t,e){super(t,e),this._isTransitioning=!1,this._triggerArray=[];const i=we.find(vi);for(const t of i){const e=we.getSelectorFromElement(t),i=we.find(e).filter((t=>t===this._element));null!==e&&i.length&&this._triggerArray.push(t)}this._initializeChildren(),this._config.parent||this._addAriaAndCollapsedClass(this._triggerArray,this._isShown()),this._config.toggle&&this.toggle()}static get Default(){return yi}static get DefaultType(){return wi}static get NAME(){return"collapse"}toggle(){this._isShown()?this.hide():this.show()}show(){if(this._isTransitioning||this._isShown())return;let t=[];if(this._config.parent&&(t=this._getFirstLevelChildren(".collapse.show, .collapse.collapsing").filter((t=>t!==this._element)).map((t=>Ei.getOrCreateInstance(t,{toggle:!1})))),t.length&&t[0]._isTransitioning)return;if(fe.trigger(this._element,hi).defaultPrevented)return;for(const e of t)e.hide();const e=this._getDimension();this._element.classList.remove(gi),this._element.classList.add(_i),this._element.style[e]=0,this._addAriaAndCollapsedClass(this._triggerArray,!0),this._isTransitioning=!0;const i=`scroll${e[0].toUpperCase()+e.slice(1)}`;this._queueCallback((()=>{this._isTransitioning=!1,this._element.classList.remove(_i),this._element.classList.add(gi,mi),this._element.style[e]="",fe.trigger(this._element,di)}),this._element,!0),this._element.style[e]=`${this._element[i]}px`}hide(){if(this._isTransitioning||!this._isShown())return;if(fe.trigger(this._element,ui).defaultPrevented)return;const t=this._getDimension();this._element.style[t]=`${this._element.getBoundingClientRect()[t]}px`,qt(this._element),this._element.classList.add(_i),this._element.classList.remove(gi,mi);for(const t of this._triggerArray){const e=we.getElementFromSelector(t);e&&!this._isShown(e)&&this._addAriaAndCollapsedClass([t],!1)}this._isTransitioning=!0,this._element.style[t]="",this._queueCallback((()=>{this._isTransitioning=!1,this._element.classList.remove(_i),this._element.classList.add(gi),fe.trigger(this._element,fi)}),this._element,!0)}_isShown(t=this._element){return t.classList.contains(mi)}_configAfterMerge(t){return t.toggle=Boolean(t.toggle),t.parent=Ht(t.parent),t}_getDimension(){return this._element.classList.contains("collapse-horizontal")?"width":"height"}_initializeChildren(){if(!this._config.parent)return;const t=this._getFirstLevelChildren(vi);for(const e of t){const t=we.getElementFromSelector(e);t&&this._addAriaAndCollapsedClass([e],this._isShown(t))}}_getFirstLevelChildren(t){const e=we.find(bi,this._config.parent);return we.find(t,this._config.parent).filter((t=>!e.includes(t)))}_addAriaAndCollapsedClass(t,e){if(t.length)for(const i of t)i.classList.toggle("collapsed",!e),i.setAttribute("aria-expanded",e)}static jQueryInterface(t){const e={};return"string"==typeof t&&/show|hide/.test(t)&&(e.toggle=!1),this.each((function(){const i=Ei.getOrCreateInstance(this,e);if("string"==typeof t){if(void 0===i[t])throw new TypeError(`No method named "${t}"`);i[t]()}}))}}fe.on(document,pi,vi,(function(t){("A"===t.target.tagName||t.delegateTarget&&"A"===t.delegateTarget.tagName)&&t.preventDefault();for(const t of we.getMultipleElementsFromSelector(this))Ei.getOrCreateInstance(t,{toggle:!1}).toggle()})),Qt(Ei);const Ai="dropdown",Ti=".bs.dropdown",Ci=".data-api",Oi="ArrowUp",xi="ArrowDown",ki=`hide${Ti}`,Li=`hidden${Ti}`,Si=`show${Ti}`,Di=`shown${Ti}`,$i=`click${Ti}${Ci}`,Ii=`keydown${Ti}${Ci}`,Ni=`keyup${Ti}${Ci}`,Pi="show",Mi='[data-bs-toggle="dropdown"]:not(.disabled):not(:disabled)',ji=`${Mi}.${Pi}`,Fi=".dropdown-menu",Hi=Kt()?"top-end":"top-start",Bi=Kt()?"top-start":"top-end",Wi=Kt()?"bottom-end":"bottom-start",zi=Kt()?"bottom-start":"bottom-end",Ri=Kt()?"left-start":"right-start",qi=Kt()?"right-start":"left-start",Vi={autoClose:!0,boundary:"clippingParents",display:"dynamic",offset:[0,2],popperConfig:null,reference:"toggle"},Yi={autoClose:"(boolean|string)",boundary:"(string|element)",display:"string",offset:"(array|string|function)",popperConfig:"(null|object|function)",reference:"(string|element|object)"};class Ki extends ve{constructor(t,e){super(t,e),this._popper=null,this._parent=this._element.parentNode,this._menu=we.next(this._element,Fi)[0]||we.prev(this._element,Fi)[0]||we.findOne(Fi,this._parent),this._inNavbar=this._detectNavbar()}static get Default(){return Vi}static get DefaultType(){return Yi}static get NAME(){return Ai}toggle(){return this._isShown()?this.hide():this.show()}show(){if(Wt(this._element)||this._isShown())return;const t={relatedTarget:this._element};if(!fe.trigger(this._element,Si,t).defaultPrevented){if(this._createPopper(),"ontouchstart"in document.documentElement&&!this._parent.closest(".navbar-nav"))for(const t of[].concat(...document.body.children))fe.on(t,"mouseover",Rt);this._element.focus(),this._element.setAttribute("aria-expanded",!0),this._menu.classList.add(Pi),this._element.classList.add(Pi),fe.trigger(this._element,Di,t)}}hide(){if(Wt(this._element)||!this._isShown())return;const t={relatedTarget:this._element};this._completeHide(t)}dispose(){this._popper&&this._popper.destroy(),super.dispose()}update(){this._inNavbar=this._detectNavbar(),this._popper&&this._popper.update()}_completeHide(t){if(!fe.trigger(this._element,ki,t).defaultPrevented){if("ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))fe.off(t,"mouseover",Rt);this._popper&&this._popper.destroy(),this._menu.classList.remove(Pi),this._element.classList.remove(Pi),this._element.setAttribute("aria-expanded","false"),_e.removeDataAttribute(this._menu,"popper"),fe.trigger(this._element,Li,t)}}_getConfig(t){if("object"==typeof(t=super._getConfig(t)).reference&&!Ft(t.reference)&&"function"!=typeof t.reference.getBoundingClientRect)throw new TypeError(`${Ai.toUpperCase()}: Option "reference" provided type "object" without a required "getBoundingClientRect" method.`);return t}_createPopper(){if(void 0===e)throw new TypeError("Bootstrap's dropdowns require Popper (https://popper.js.org)");let t=this._element;"parent"===this._config.reference?t=this._parent:Ft(this._config.reference)?t=Ht(this._config.reference):"object"==typeof this._config.reference&&(t=this._config.reference);const i=this._getPopperConfig();this._popper=Dt(t,this._menu,i)}_isShown(){return this._menu.classList.contains(Pi)}_getPlacement(){const t=this._parent;if(t.classList.contains("dropend"))return Ri;if(t.classList.contains("dropstart"))return qi;if(t.classList.contains("dropup-center"))return"top";if(t.classList.contains("dropdown-center"))return"bottom";const e="end"===getComputedStyle(this._menu).getPropertyValue("--bs-position").trim();return t.classList.contains("dropup")?e?Bi:Hi:e?zi:Wi}_detectNavbar(){return null!==this._element.closest(".navbar")}_getOffset(){const{offset:t}=this._config;return"string"==typeof t?t.split(",").map((t=>Number.parseInt(t,10))):"function"==typeof t?e=>t(e,this._element):t}_getPopperConfig(){const t={placement:this._getPlacement(),modifiers:[{name:"preventOverflow",options:{boundary:this._config.boundary}},{name:"offset",options:{offset:this._getOffset()}}]};return(this._inNavbar||"static"===this._config.display)&&(_e.setDataAttribute(this._menu,"popper","static"),t.modifiers=[{name:"applyStyles",enabled:!1}]),{...t,...Xt(this._config.popperConfig,[t])}}_selectMenuItem({key:t,target:e}){const i=we.find(".dropdown-menu .dropdown-item:not(.disabled):not(:disabled)",this._menu).filter((t=>Bt(t)));i.length&&Gt(i,e,t===xi,!i.includes(e)).focus()}static jQueryInterface(t){return this.each((function(){const e=Ki.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}static clearMenus(t){if(2===t.button||"keyup"===t.type&&"Tab"!==t.key)return;const e=we.find(ji);for(const i of e){const e=Ki.getInstance(i);if(!e||!1===e._config.autoClose)continue;const n=t.composedPath(),s=n.includes(e._menu);if(n.includes(e._element)||"inside"===e._config.autoClose&&!s||"outside"===e._config.autoClose&&s)continue;if(e._menu.contains(t.target)&&("keyup"===t.type&&"Tab"===t.key||/input|select|option|textarea|form/i.test(t.target.tagName)))continue;const o={relatedTarget:e._element};"click"===t.type&&(o.clickEvent=t),e._completeHide(o)}}static dataApiKeydownHandler(t){const e=/input|textarea/i.test(t.target.tagName),i="Escape"===t.key,n=[Oi,xi].includes(t.key);if(!n&&!i)return;if(e&&!i)return;t.preventDefault();const s=this.matches(Mi)?this:we.prev(this,Mi)[0]||we.next(this,Mi)[0]||we.findOne(Mi,t.delegateTarget.parentNode),o=Ki.getOrCreateInstance(s);if(n)return t.stopPropagation(),o.show(),void o._selectMenuItem(t);o._isShown()&&(t.stopPropagation(),o.hide(),s.focus())}}fe.on(document,Ii,Mi,Ki.dataApiKeydownHandler),fe.on(document,Ii,Fi,Ki.dataApiKeydownHandler),fe.on(document,$i,Ki.clearMenus),fe.on(document,Ni,Ki.clearMenus),fe.on(document,$i,Mi,(function(t){t.preventDefault(),Ki.getOrCreateInstance(this).toggle()})),Qt(Ki);const Qi="backdrop",Xi="show",Ui=`mousedown.bs.${Qi}`,Gi={className:"modal-backdrop",clickCallback:null,isAnimated:!1,isVisible:!0,rootElement:"body"},Ji={className:"string",clickCallback:"(function|null)",isAnimated:"boolean",isVisible:"boolean",rootElement:"(element|string)"};class Zi extends be{constructor(t){super(),this._config=this._getConfig(t),this._isAppended=!1,this._element=null}static get Default(){return Gi}static get DefaultType(){return Ji}static get NAME(){return Qi}show(t){if(!this._config.isVisible)return void Xt(t);this._append();const e=this._getElement();this._config.isAnimated&&qt(e),e.classList.add(Xi),this._emulateAnimation((()=>{Xt(t)}))}hide(t){this._config.isVisible?(this._getElement().classList.remove(Xi),this._emulateAnimation((()=>{this.dispose(),Xt(t)}))):Xt(t)}dispose(){this._isAppended&&(fe.off(this._element,Ui),this._element.remove(),this._isAppended=!1)}_getElement(){if(!this._element){const t=document.createElement("div");t.className=this._config.className,this._config.isAnimated&&t.classList.add("fade"),this._element=t}return this._element}_configAfterMerge(t){return t.rootElement=Ht(t.rootElement),t}_append(){if(this._isAppended)return;const t=this._getElement();this._config.rootElement.append(t),fe.on(t,Ui,(()=>{Xt(this._config.clickCallback)})),this._isAppended=!0}_emulateAnimation(t){Ut(t,this._getElement(),this._config.isAnimated)}}const tn=".bs.focustrap",en=`focusin${tn}`,nn=`keydown.tab${tn}`,sn="backward",on={autofocus:!0,trapElement:null},rn={autofocus:"boolean",trapElement:"element"};class an extends be{constructor(t){super(),this._config=this._getConfig(t),this._isActive=!1,this._lastTabNavDirection=null}static get Default(){return on}static get DefaultType(){return rn}static get NAME(){return"focustrap"}activate(){this._isActive||(this._config.autofocus&&this._config.trapElement.focus(),fe.off(document,tn),fe.on(document,en,(t=>this._handleFocusin(t))),fe.on(document,nn,(t=>this._handleKeydown(t))),this._isActive=!0)}deactivate(){this._isActive&&(this._isActive=!1,fe.off(document,tn))}_handleFocusin(t){const{trapElement:e}=this._config;if(t.target===document||t.target===e||e.contains(t.target))return;const i=we.focusableChildren(e);0===i.length?e.focus():this._lastTabNavDirection===sn?i[i.length-1].focus():i[0].focus()}_handleKeydown(t){"Tab"===t.key&&(this._lastTabNavDirection=t.shiftKey?sn:"forward")}}const ln=".fixed-top, .fixed-bottom, .is-fixed, .sticky-top",cn=".sticky-top",hn="padding-right",dn="margin-right";class un{constructor(){this._element=document.body}getWidth(){const t=document.documentElement.clientWidth;return Math.abs(window.innerWidth-t)}hide(){const t=this.getWidth();this._disableOverFlow(),this._setElementAttributes(this._element,hn,(e=>e+t)),this._setElementAttributes(ln,hn,(e=>e+t)),this._setElementAttributes(cn,dn,(e=>e-t))}reset(){this._resetElementAttributes(this._element,"overflow"),this._resetElementAttributes(this._element,hn),this._resetElementAttributes(ln,hn),this._resetElementAttributes(cn,dn)}isOverflowing(){return this.getWidth()>0}_disableOverFlow(){this._saveInitialAttribute(this._element,"overflow"),this._element.style.overflow="hidden"}_setElementAttributes(t,e,i){const n=this.getWidth();this._applyManipulationCallback(t,(t=>{if(t!==this._element&&window.innerWidth>t.clientWidth+n)return;this._saveInitialAttribute(t,e);const s=window.getComputedStyle(t).getPropertyValue(e);t.style.setProperty(e,`${i(Number.parseFloat(s))}px`)}))}_saveInitialAttribute(t,e){const i=t.style.getPropertyValue(e);i&&_e.setDataAttribute(t,e,i)}_resetElementAttributes(t,e){this._applyManipulationCallback(t,(t=>{const i=_e.getDataAttribute(t,e);null!==i?(_e.removeDataAttribute(t,e),t.style.setProperty(e,i)):t.style.removeProperty(e)}))}_applyManipulationCallback(t,e){if(Ft(t))e(t);else for(const i of we.find(t,this._element))e(i)}}const fn=".bs.modal",pn=`hide${fn}`,mn=`hidePrevented${fn}`,gn=`hidden${fn}`,_n=`show${fn}`,bn=`shown${fn}`,vn=`resize${fn}`,yn=`click.dismiss${fn}`,wn=`mousedown.dismiss${fn}`,En=`keydown.dismiss${fn}`,An=`click${fn}.data-api`,Tn="modal-open",Cn="show",On="modal-static",xn={backdrop:!0,focus:!0,keyboard:!0},kn={backdrop:"(boolean|string)",focus:"boolean",keyboard:"boolean"};class Ln extends ve{constructor(t,e){super(t,e),this._dialog=we.findOne(".modal-dialog",this._element),this._backdrop=this._initializeBackDrop(),this._focustrap=this._initializeFocusTrap(),this._isShown=!1,this._isTransitioning=!1,this._scrollBar=new un,this._addEventListeners()}static get Default(){return xn}static get DefaultType(){return kn}static get NAME(){return"modal"}toggle(t){return this._isShown?this.hide():this.show(t)}show(t){this._isShown||this._isTransitioning||fe.trigger(this._element,_n,{relatedTarget:t}).defaultPrevented||(this._isShown=!0,this._isTransitioning=!0,this._scrollBar.hide(),document.body.classList.add(Tn),this._adjustDialog(),this._backdrop.show((()=>this._showElement(t))))}hide(){this._isShown&&!this._isTransitioning&&(fe.trigger(this._element,pn).defaultPrevented||(this._isShown=!1,this._isTransitioning=!0,this._focustrap.deactivate(),this._element.classList.remove(Cn),this._queueCallback((()=>this._hideModal()),this._element,this._isAnimated())))}dispose(){fe.off(window,fn),fe.off(this._dialog,fn),this._backdrop.dispose(),this._focustrap.deactivate(),super.dispose()}handleUpdate(){this._adjustDialog()}_initializeBackDrop(){return new Zi({isVisible:Boolean(this._config.backdrop),isAnimated:this._isAnimated()})}_initializeFocusTrap(){return new an({trapElement:this._element})}_showElement(t){document.body.contains(this._element)||document.body.append(this._element),this._element.style.display="block",this._element.removeAttribute("aria-hidden"),this._element.setAttribute("aria-modal",!0),this._element.setAttribute("role","dialog"),this._element.scrollTop=0;const e=we.findOne(".modal-body",this._dialog);e&&(e.scrollTop=0),qt(this._element),this._element.classList.add(Cn),this._queueCallback((()=>{this._config.focus&&this._focustrap.activate(),this._isTransitioning=!1,fe.trigger(this._element,bn,{relatedTarget:t})}),this._dialog,this._isAnimated())}_addEventListeners(){fe.on(this._element,En,(t=>{"Escape"===t.key&&(this._config.keyboard?this.hide():this._triggerBackdropTransition())})),fe.on(window,vn,(()=>{this._isShown&&!this._isTransitioning&&this._adjustDialog()})),fe.on(this._element,wn,(t=>{fe.one(this._element,yn,(e=>{this._element===t.target&&this._element===e.target&&("static"!==this._config.backdrop?this._config.backdrop&&this.hide():this._triggerBackdropTransition())}))}))}_hideModal(){this._element.style.display="none",this._element.setAttribute("aria-hidden",!0),this._element.removeAttribute("aria-modal"),this._element.removeAttribute("role"),this._isTransitioning=!1,this._backdrop.hide((()=>{document.body.classList.remove(Tn),this._resetAdjustments(),this._scrollBar.reset(),fe.trigger(this._element,gn)}))}_isAnimated(){return this._element.classList.contains("fade")}_triggerBackdropTransition(){if(fe.trigger(this._element,mn).defaultPrevented)return;const t=this._element.scrollHeight>document.documentElement.clientHeight,e=this._element.style.overflowY;"hidden"===e||this._element.classList.contains(On)||(t||(this._element.style.overflowY="hidden"),this._element.classList.add(On),this._queueCallback((()=>{this._element.classList.remove(On),this._queueCallback((()=>{this._element.style.overflowY=e}),this._dialog)}),this._dialog),this._element.focus())}_adjustDialog(){const t=this._element.scrollHeight>document.documentElement.clientHeight,e=this._scrollBar.getWidth(),i=e>0;if(i&&!t){const t=Kt()?"paddingLeft":"paddingRight";this._element.style[t]=`${e}px`}if(!i&&t){const t=Kt()?"paddingRight":"paddingLeft";this._element.style[t]=`${e}px`}}_resetAdjustments(){this._element.style.paddingLeft="",this._element.style.paddingRight=""}static jQueryInterface(t,e){return this.each((function(){const i=Ln.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===i[t])throw new TypeError(`No method named "${t}"`);i[t](e)}}))}}fe.on(document,An,'[data-bs-toggle="modal"]',(function(t){const e=we.getElementFromSelector(this);["A","AREA"].includes(this.tagName)&&t.preventDefault(),fe.one(e,_n,(t=>{t.defaultPrevented||fe.one(e,gn,(()=>{Bt(this)&&this.focus()}))}));const i=we.findOne(".modal.show");i&&Ln.getInstance(i).hide(),Ln.getOrCreateInstance(e).toggle(this)})),Ee(Ln),Qt(Ln);const Sn=".bs.offcanvas",Dn=".data-api",$n=`load${Sn}${Dn}`,In="show",Nn="showing",Pn="hiding",Mn=".offcanvas.show",jn=`show${Sn}`,Fn=`shown${Sn}`,Hn=`hide${Sn}`,Bn=`hidePrevented${Sn}`,Wn=`hidden${Sn}`,zn=`resize${Sn}`,Rn=`click${Sn}${Dn}`,qn=`keydown.dismiss${Sn}`,Vn={backdrop:!0,keyboard:!0,scroll:!1},Yn={backdrop:"(boolean|string)",keyboard:"boolean",scroll:"boolean"};class Kn extends ve{constructor(t,e){super(t,e),this._isShown=!1,this._backdrop=this._initializeBackDrop(),this._focustrap=this._initializeFocusTrap(),this._addEventListeners()}static get Default(){return Vn}static get DefaultType(){return Yn}static get NAME(){return"offcanvas"}toggle(t){return this._isShown?this.hide():this.show(t)}show(t){this._isShown||fe.trigger(this._element,jn,{relatedTarget:t}).defaultPrevented||(this._isShown=!0,this._backdrop.show(),this._config.scroll||(new un).hide(),this._element.setAttribute("aria-modal",!0),this._element.setAttribute("role","dialog"),this._element.classList.add(Nn),this._queueCallback((()=>{this._config.scroll&&!this._config.backdrop||this._focustrap.activate(),this._element.classList.add(In),this._element.classList.remove(Nn),fe.trigger(this._element,Fn,{relatedTarget:t})}),this._element,!0))}hide(){this._isShown&&(fe.trigger(this._element,Hn).defaultPrevented||(this._focustrap.deactivate(),this._element.blur(),this._isShown=!1,this._element.classList.add(Pn),this._backdrop.hide(),this._queueCallback((()=>{this._element.classList.remove(In,Pn),this._element.removeAttribute("aria-modal"),this._element.removeAttribute("role"),this._config.scroll||(new un).reset(),fe.trigger(this._element,Wn)}),this._element,!0)))}dispose(){this._backdrop.dispose(),this._focustrap.deactivate(),super.dispose()}_initializeBackDrop(){const t=Boolean(this._config.backdrop);return new Zi({className:"offcanvas-backdrop",isVisible:t,isAnimated:!0,rootElement:this._element.parentNode,clickCallback:t?()=>{"static"!==this._config.backdrop?this.hide():fe.trigger(this._element,Bn)}:null})}_initializeFocusTrap(){return new an({trapElement:this._element})}_addEventListeners(){fe.on(this._element,qn,(t=>{"Escape"===t.key&&(this._config.keyboard?this.hide():fe.trigger(this._element,Bn))}))}static jQueryInterface(t){return this.each((function(){const e=Kn.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}fe.on(document,Rn,'[data-bs-toggle="offcanvas"]',(function(t){const e=we.getElementFromSelector(this);if(["A","AREA"].includes(this.tagName)&&t.preventDefault(),Wt(this))return;fe.one(e,Wn,(()=>{Bt(this)&&this.focus()}));const i=we.findOne(Mn);i&&i!==e&&Kn.getInstance(i).hide(),Kn.getOrCreateInstance(e).toggle(this)})),fe.on(window,$n,(()=>{for(const t of we.find(Mn))Kn.getOrCreateInstance(t).show()})),fe.on(window,zn,(()=>{for(const t of we.find("[aria-modal][class*=show][class*=offcanvas-]"))"fixed"!==getComputedStyle(t).position&&Kn.getOrCreateInstance(t).hide()})),Ee(Kn),Qt(Kn);const Qn={"*":["class","dir","id","lang","role",/^aria-[\w-]*$/i],a:["target","href","title","rel"],area:[],b:[],br:[],col:[],code:[],dd:[],div:[],dl:[],dt:[],em:[],hr:[],h1:[],h2:[],h3:[],h4:[],h5:[],h6:[],i:[],img:["src","srcset","alt","title","width","height"],li:[],ol:[],p:[],pre:[],s:[],small:[],span:[],sub:[],sup:[],strong:[],u:[],ul:[]},Xn=new Set(["background","cite","href","itemtype","longdesc","poster","src","xlink:href"]),Un=/^(?!javascript:)(?:[a-z0-9+.-]+:|[^&:/?#]*(?:[/?#]|$))/i,Gn=(t,e)=>{const i=t.nodeName.toLowerCase();return e.includes(i)?!Xn.has(i)||Boolean(Un.test(t.nodeValue)):e.filter((t=>t instanceof RegExp)).some((t=>t.test(i)))},Jn={allowList:Qn,content:{},extraClass:"",html:!1,sanitize:!0,sanitizeFn:null,template:"
"},Zn={allowList:"object",content:"object",extraClass:"(string|function)",html:"boolean",sanitize:"boolean",sanitizeFn:"(null|function)",template:"string"},ts={entry:"(string|element|function|null)",selector:"(string|element)"};class es extends be{constructor(t){super(),this._config=this._getConfig(t)}static get Default(){return Jn}static get DefaultType(){return Zn}static get NAME(){return"TemplateFactory"}getContent(){return Object.values(this._config.content).map((t=>this._resolvePossibleFunction(t))).filter(Boolean)}hasContent(){return this.getContent().length>0}changeContent(t){return this._checkContent(t),this._config.content={...this._config.content,...t},this}toHtml(){const t=document.createElement("div");t.innerHTML=this._maybeSanitize(this._config.template);for(const[e,i]of Object.entries(this._config.content))this._setContent(t,i,e);const e=t.children[0],i=this._resolvePossibleFunction(this._config.extraClass);return i&&e.classList.add(...i.split(" ")),e}_typeCheckConfig(t){super._typeCheckConfig(t),this._checkContent(t.content)}_checkContent(t){for(const[e,i]of Object.entries(t))super._typeCheckConfig({selector:e,entry:i},ts)}_setContent(t,e,i){const n=we.findOne(i,t);n&&((e=this._resolvePossibleFunction(e))?Ft(e)?this._putElementInTemplate(Ht(e),n):this._config.html?n.innerHTML=this._maybeSanitize(e):n.textContent=e:n.remove())}_maybeSanitize(t){return this._config.sanitize?function(t,e,i){if(!t.length)return t;if(i&&"function"==typeof i)return i(t);const n=(new window.DOMParser).parseFromString(t,"text/html"),s=[].concat(...n.body.querySelectorAll("*"));for(const t of s){const i=t.nodeName.toLowerCase();if(!Object.keys(e).includes(i)){t.remove();continue}const n=[].concat(...t.attributes),s=[].concat(e["*"]||[],e[i]||[]);for(const e of n)Gn(e,s)||t.removeAttribute(e.nodeName)}return n.body.innerHTML}(t,this._config.allowList,this._config.sanitizeFn):t}_resolvePossibleFunction(t){return Xt(t,[this])}_putElementInTemplate(t,e){if(this._config.html)return e.innerHTML="",void e.append(t);e.textContent=t.textContent}}const is=new Set(["sanitize","allowList","sanitizeFn"]),ns="fade",ss="show",os=".tooltip-inner",rs=".modal",as="hide.bs.modal",ls="hover",cs="focus",hs={AUTO:"auto",TOP:"top",RIGHT:Kt()?"left":"right",BOTTOM:"bottom",LEFT:Kt()?"right":"left"},ds={allowList:Qn,animation:!0,boundary:"clippingParents",container:!1,customClass:"",delay:0,fallbackPlacements:["top","right","bottom","left"],html:!1,offset:[0,6],placement:"top",popperConfig:null,sanitize:!0,sanitizeFn:null,selector:!1,template:'',title:"",trigger:"hover focus"},us={allowList:"object",animation:"boolean",boundary:"(string|element)",container:"(string|element|boolean)",customClass:"(string|function)",delay:"(number|object)",fallbackPlacements:"array",html:"boolean",offset:"(array|string|function)",placement:"(string|function)",popperConfig:"(null|object|function)",sanitize:"boolean",sanitizeFn:"(null|function)",selector:"(string|boolean)",template:"string",title:"(string|element|function)",trigger:"string"};class fs extends ve{constructor(t,i){if(void 0===e)throw new TypeError("Bootstrap's tooltips require Popper (https://popper.js.org)");super(t,i),this._isEnabled=!0,this._timeout=0,this._isHovered=null,this._activeTrigger={},this._popper=null,this._templateFactory=null,this._newContent=null,this.tip=null,this._setListeners(),this._config.selector||this._fixTitle()}static get Default(){return ds}static get DefaultType(){return us}static get NAME(){return"tooltip"}enable(){this._isEnabled=!0}disable(){this._isEnabled=!1}toggleEnabled(){this._isEnabled=!this._isEnabled}toggle(){this._isEnabled&&(this._activeTrigger.click=!this._activeTrigger.click,this._isShown()?this._leave():this._enter())}dispose(){clearTimeout(this._timeout),fe.off(this._element.closest(rs),as,this._hideModalHandler),this._element.getAttribute("data-bs-original-title")&&this._element.setAttribute("title",this._element.getAttribute("data-bs-original-title")),this._disposePopper(),super.dispose()}show(){if("none"===this._element.style.display)throw new Error("Please use show on visible elements");if(!this._isWithContent()||!this._isEnabled)return;const t=fe.trigger(this._element,this.constructor.eventName("show")),e=(zt(this._element)||this._element.ownerDocument.documentElement).contains(this._element);if(t.defaultPrevented||!e)return;this._disposePopper();const i=this._getTipElement();this._element.setAttribute("aria-describedby",i.getAttribute("id"));const{container:n}=this._config;if(this._element.ownerDocument.documentElement.contains(this.tip)||(n.append(i),fe.trigger(this._element,this.constructor.eventName("inserted"))),this._popper=this._createPopper(i),i.classList.add(ss),"ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))fe.on(t,"mouseover",Rt);this._queueCallback((()=>{fe.trigger(this._element,this.constructor.eventName("shown")),!1===this._isHovered&&this._leave(),this._isHovered=!1}),this.tip,this._isAnimated())}hide(){if(this._isShown()&&!fe.trigger(this._element,this.constructor.eventName("hide")).defaultPrevented){if(this._getTipElement().classList.remove(ss),"ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))fe.off(t,"mouseover",Rt);this._activeTrigger.click=!1,this._activeTrigger[cs]=!1,this._activeTrigger[ls]=!1,this._isHovered=null,this._queueCallback((()=>{this._isWithActiveTrigger()||(this._isHovered||this._disposePopper(),this._element.removeAttribute("aria-describedby"),fe.trigger(this._element,this.constructor.eventName("hidden")))}),this.tip,this._isAnimated())}}update(){this._popper&&this._popper.update()}_isWithContent(){return Boolean(this._getTitle())}_getTipElement(){return this.tip||(this.tip=this._createTipElement(this._newContent||this._getContentForTemplate())),this.tip}_createTipElement(t){const e=this._getTemplateFactory(t).toHtml();if(!e)return null;e.classList.remove(ns,ss),e.classList.add(`bs-${this.constructor.NAME}-auto`);const i=(t=>{do{t+=Math.floor(1e6*Math.random())}while(document.getElementById(t));return t})(this.constructor.NAME).toString();return e.setAttribute("id",i),this._isAnimated()&&e.classList.add(ns),e}setContent(t){this._newContent=t,this._isShown()&&(this._disposePopper(),this.show())}_getTemplateFactory(t){return this._templateFactory?this._templateFactory.changeContent(t):this._templateFactory=new es({...this._config,content:t,extraClass:this._resolvePossibleFunction(this._config.customClass)}),this._templateFactory}_getContentForTemplate(){return{[os]:this._getTitle()}}_getTitle(){return this._resolvePossibleFunction(this._config.title)||this._element.getAttribute("data-bs-original-title")}_initializeOnDelegatedTarget(t){return this.constructor.getOrCreateInstance(t.delegateTarget,this._getDelegateConfig())}_isAnimated(){return this._config.animation||this.tip&&this.tip.classList.contains(ns)}_isShown(){return this.tip&&this.tip.classList.contains(ss)}_createPopper(t){const e=Xt(this._config.placement,[this,t,this._element]),i=hs[e.toUpperCase()];return Dt(this._element,t,this._getPopperConfig(i))}_getOffset(){const{offset:t}=this._config;return"string"==typeof t?t.split(",").map((t=>Number.parseInt(t,10))):"function"==typeof t?e=>t(e,this._element):t}_resolvePossibleFunction(t){return Xt(t,[this._element])}_getPopperConfig(t){const e={placement:t,modifiers:[{name:"flip",options:{fallbackPlacements:this._config.fallbackPlacements}},{name:"offset",options:{offset:this._getOffset()}},{name:"preventOverflow",options:{boundary:this._config.boundary}},{name:"arrow",options:{element:`.${this.constructor.NAME}-arrow`}},{name:"preSetPlacement",enabled:!0,phase:"beforeMain",fn:t=>{this._getTipElement().setAttribute("data-popper-placement",t.state.placement)}}]};return{...e,...Xt(this._config.popperConfig,[e])}}_setListeners(){const t=this._config.trigger.split(" ");for(const e of t)if("click"===e)fe.on(this._element,this.constructor.eventName("click"),this._config.selector,(t=>{this._initializeOnDelegatedTarget(t).toggle()}));else if("manual"!==e){const t=e===ls?this.constructor.eventName("mouseenter"):this.constructor.eventName("focusin"),i=e===ls?this.constructor.eventName("mouseleave"):this.constructor.eventName("focusout");fe.on(this._element,t,this._config.selector,(t=>{const e=this._initializeOnDelegatedTarget(t);e._activeTrigger["focusin"===t.type?cs:ls]=!0,e._enter()})),fe.on(this._element,i,this._config.selector,(t=>{const e=this._initializeOnDelegatedTarget(t);e._activeTrigger["focusout"===t.type?cs:ls]=e._element.contains(t.relatedTarget),e._leave()}))}this._hideModalHandler=()=>{this._element&&this.hide()},fe.on(this._element.closest(rs),as,this._hideModalHandler)}_fixTitle(){const t=this._element.getAttribute("title");t&&(this._element.getAttribute("aria-label")||this._element.textContent.trim()||this._element.setAttribute("aria-label",t),this._element.setAttribute("data-bs-original-title",t),this._element.removeAttribute("title"))}_enter(){this._isShown()||this._isHovered?this._isHovered=!0:(this._isHovered=!0,this._setTimeout((()=>{this._isHovered&&this.show()}),this._config.delay.show))}_leave(){this._isWithActiveTrigger()||(this._isHovered=!1,this._setTimeout((()=>{this._isHovered||this.hide()}),this._config.delay.hide))}_setTimeout(t,e){clearTimeout(this._timeout),this._timeout=setTimeout(t,e)}_isWithActiveTrigger(){return Object.values(this._activeTrigger).includes(!0)}_getConfig(t){const e=_e.getDataAttributes(this._element);for(const t of Object.keys(e))is.has(t)&&delete e[t];return t={...e,..."object"==typeof t&&t?t:{}},t=this._mergeConfigObj(t),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}_configAfterMerge(t){return t.container=!1===t.container?document.body:Ht(t.container),"number"==typeof t.delay&&(t.delay={show:t.delay,hide:t.delay}),"number"==typeof t.title&&(t.title=t.title.toString()),"number"==typeof t.content&&(t.content=t.content.toString()),t}_getDelegateConfig(){const t={};for(const[e,i]of Object.entries(this._config))this.constructor.Default[e]!==i&&(t[e]=i);return t.selector=!1,t.trigger="manual",t}_disposePopper(){this._popper&&(this._popper.destroy(),this._popper=null),this.tip&&(this.tip.remove(),this.tip=null)}static jQueryInterface(t){return this.each((function(){const e=fs.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}}Qt(fs);const ps=".popover-header",ms=".popover-body",gs={...fs.Default,content:"",offset:[0,8],placement:"right",template:'',trigger:"click"},_s={...fs.DefaultType,content:"(null|string|element|function)"};class bs extends fs{static get Default(){return gs}static get DefaultType(){return _s}static get NAME(){return"popover"}_isWithContent(){return this._getTitle()||this._getContent()}_getContentForTemplate(){return{[ps]:this._getTitle(),[ms]:this._getContent()}}_getContent(){return this._resolvePossibleFunction(this._config.content)}static jQueryInterface(t){return this.each((function(){const e=bs.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}}Qt(bs);const vs=".bs.scrollspy",ys=`activate${vs}`,ws=`click${vs}`,Es=`load${vs}.data-api`,As="active",Ts="[href]",Cs=".nav-link",Os=`${Cs}, .nav-item > ${Cs}, .list-group-item`,xs={offset:null,rootMargin:"0px 0px -25%",smoothScroll:!1,target:null,threshold:[.1,.5,1]},ks={offset:"(number|null)",rootMargin:"string",smoothScroll:"boolean",target:"element",threshold:"array"};class Ls extends ve{constructor(t,e){super(t,e),this._targetLinks=new Map,this._observableSections=new Map,this._rootElement="visible"===getComputedStyle(this._element).overflowY?null:this._element,this._activeTarget=null,this._observer=null,this._previousScrollData={visibleEntryTop:0,parentScrollTop:0},this.refresh()}static get Default(){return xs}static get DefaultType(){return ks}static get NAME(){return"scrollspy"}refresh(){this._initializeTargetsAndObservables(),this._maybeEnableSmoothScroll(),this._observer?this._observer.disconnect():this._observer=this._getNewObserver();for(const t of this._observableSections.values())this._observer.observe(t)}dispose(){this._observer.disconnect(),super.dispose()}_configAfterMerge(t){return t.target=Ht(t.target)||document.body,t.rootMargin=t.offset?`${t.offset}px 0px -30%`:t.rootMargin,"string"==typeof t.threshold&&(t.threshold=t.threshold.split(",").map((t=>Number.parseFloat(t)))),t}_maybeEnableSmoothScroll(){this._config.smoothScroll&&(fe.off(this._config.target,ws),fe.on(this._config.target,ws,Ts,(t=>{const e=this._observableSections.get(t.target.hash);if(e){t.preventDefault();const i=this._rootElement||window,n=e.offsetTop-this._element.offsetTop;if(i.scrollTo)return void i.scrollTo({top:n,behavior:"smooth"});i.scrollTop=n}})))}_getNewObserver(){const t={root:this._rootElement,threshold:this._config.threshold,rootMargin:this._config.rootMargin};return new IntersectionObserver((t=>this._observerCallback(t)),t)}_observerCallback(t){const e=t=>this._targetLinks.get(`#${t.target.id}`),i=t=>{this._previousScrollData.visibleEntryTop=t.target.offsetTop,this._process(e(t))},n=(this._rootElement||document.documentElement).scrollTop,s=n>=this._previousScrollData.parentScrollTop;this._previousScrollData.parentScrollTop=n;for(const o of t){if(!o.isIntersecting){this._activeTarget=null,this._clearActiveClass(e(o));continue}const t=o.target.offsetTop>=this._previousScrollData.visibleEntryTop;if(s&&t){if(i(o),!n)return}else s||t||i(o)}}_initializeTargetsAndObservables(){this._targetLinks=new Map,this._observableSections=new Map;const t=we.find(Ts,this._config.target);for(const e of t){if(!e.hash||Wt(e))continue;const t=we.findOne(decodeURI(e.hash),this._element);Bt(t)&&(this._targetLinks.set(decodeURI(e.hash),e),this._observableSections.set(e.hash,t))}}_process(t){this._activeTarget!==t&&(this._clearActiveClass(this._config.target),this._activeTarget=t,t.classList.add(As),this._activateParents(t),fe.trigger(this._element,ys,{relatedTarget:t}))}_activateParents(t){if(t.classList.contains("dropdown-item"))we.findOne(".dropdown-toggle",t.closest(".dropdown")).classList.add(As);else for(const e of we.parents(t,".nav, .list-group"))for(const t of we.prev(e,Os))t.classList.add(As)}_clearActiveClass(t){t.classList.remove(As);const e=we.find(`${Ts}.${As}`,t);for(const t of e)t.classList.remove(As)}static jQueryInterface(t){return this.each((function(){const e=Ls.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}))}}fe.on(window,Es,(()=>{for(const t of we.find('[data-bs-spy="scroll"]'))Ls.getOrCreateInstance(t)})),Qt(Ls);const Ss=".bs.tab",Ds=`hide${Ss}`,$s=`hidden${Ss}`,Is=`show${Ss}`,Ns=`shown${Ss}`,Ps=`click${Ss}`,Ms=`keydown${Ss}`,js=`load${Ss}`,Fs="ArrowLeft",Hs="ArrowRight",Bs="ArrowUp",Ws="ArrowDown",zs="Home",Rs="End",qs="active",Vs="fade",Ys="show",Ks=".dropdown-toggle",Qs=`:not(${Ks})`,Xs='[data-bs-toggle="tab"], [data-bs-toggle="pill"], [data-bs-toggle="list"]',Us=`.nav-link${Qs}, .list-group-item${Qs}, [role="tab"]${Qs}, ${Xs}`,Gs=`.${qs}[data-bs-toggle="tab"], .${qs}[data-bs-toggle="pill"], .${qs}[data-bs-toggle="list"]`;class Js extends ve{constructor(t){super(t),this._parent=this._element.closest('.list-group, .nav, [role="tablist"]'),this._parent&&(this._setInitialAttributes(this._parent,this._getChildren()),fe.on(this._element,Ms,(t=>this._keydown(t))))}static get NAME(){return"tab"}show(){const t=this._element;if(this._elemIsActive(t))return;const e=this._getActiveElem(),i=e?fe.trigger(e,Ds,{relatedTarget:t}):null;fe.trigger(t,Is,{relatedTarget:e}).defaultPrevented||i&&i.defaultPrevented||(this._deactivate(e,t),this._activate(t,e))}_activate(t,e){t&&(t.classList.add(qs),this._activate(we.getElementFromSelector(t)),this._queueCallback((()=>{"tab"===t.getAttribute("role")?(t.removeAttribute("tabindex"),t.setAttribute("aria-selected",!0),this._toggleDropDown(t,!0),fe.trigger(t,Ns,{relatedTarget:e})):t.classList.add(Ys)}),t,t.classList.contains(Vs)))}_deactivate(t,e){t&&(t.classList.remove(qs),t.blur(),this._deactivate(we.getElementFromSelector(t)),this._queueCallback((()=>{"tab"===t.getAttribute("role")?(t.setAttribute("aria-selected",!1),t.setAttribute("tabindex","-1"),this._toggleDropDown(t,!1),fe.trigger(t,$s,{relatedTarget:e})):t.classList.remove(Ys)}),t,t.classList.contains(Vs)))}_keydown(t){if(![Fs,Hs,Bs,Ws,zs,Rs].includes(t.key))return;t.stopPropagation(),t.preventDefault();const e=this._getChildren().filter((t=>!Wt(t)));let i;if([zs,Rs].includes(t.key))i=e[t.key===zs?0:e.length-1];else{const n=[Hs,Ws].includes(t.key);i=Gt(e,t.target,n,!0)}i&&(i.focus({preventScroll:!0}),Js.getOrCreateInstance(i).show())}_getChildren(){return we.find(Us,this._parent)}_getActiveElem(){return this._getChildren().find((t=>this._elemIsActive(t)))||null}_setInitialAttributes(t,e){this._setAttributeIfNotExists(t,"role","tablist");for(const t of e)this._setInitialAttributesOnChild(t)}_setInitialAttributesOnChild(t){t=this._getInnerElement(t);const e=this._elemIsActive(t),i=this._getOuterElement(t);t.setAttribute("aria-selected",e),i!==t&&this._setAttributeIfNotExists(i,"role","presentation"),e||t.setAttribute("tabindex","-1"),this._setAttributeIfNotExists(t,"role","tab"),this._setInitialAttributesOnTargetPanel(t)}_setInitialAttributesOnTargetPanel(t){const e=we.getElementFromSelector(t);e&&(this._setAttributeIfNotExists(e,"role","tabpanel"),t.id&&this._setAttributeIfNotExists(e,"aria-labelledby",`${t.id}`))}_toggleDropDown(t,e){const i=this._getOuterElement(t);if(!i.classList.contains("dropdown"))return;const n=(t,n)=>{const s=we.findOne(t,i);s&&s.classList.toggle(n,e)};n(Ks,qs),n(".dropdown-menu",Ys),i.setAttribute("aria-expanded",e)}_setAttributeIfNotExists(t,e,i){t.hasAttribute(e)||t.setAttribute(e,i)}_elemIsActive(t){return t.classList.contains(qs)}_getInnerElement(t){return t.matches(Us)?t:we.findOne(Us,t)}_getOuterElement(t){return t.closest(".nav-item, .list-group-item")||t}static jQueryInterface(t){return this.each((function(){const e=Js.getOrCreateInstance(this);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}))}}fe.on(document,Ps,Xs,(function(t){["A","AREA"].includes(this.tagName)&&t.preventDefault(),Wt(this)||Js.getOrCreateInstance(this).show()})),fe.on(window,js,(()=>{for(const t of we.find(Gs))Js.getOrCreateInstance(t)})),Qt(Js);const Zs=".bs.toast",to=`mouseover${Zs}`,eo=`mouseout${Zs}`,io=`focusin${Zs}`,no=`focusout${Zs}`,so=`hide${Zs}`,oo=`hidden${Zs}`,ro=`show${Zs}`,ao=`shown${Zs}`,lo="hide",co="show",ho="showing",uo={animation:"boolean",autohide:"boolean",delay:"number"},fo={animation:!0,autohide:!0,delay:5e3};class po extends ve{constructor(t,e){super(t,e),this._timeout=null,this._hasMouseInteraction=!1,this._hasKeyboardInteraction=!1,this._setListeners()}static get Default(){return fo}static get DefaultType(){return uo}static get NAME(){return"toast"}show(){fe.trigger(this._element,ro).defaultPrevented||(this._clearTimeout(),this._config.animation&&this._element.classList.add("fade"),this._element.classList.remove(lo),qt(this._element),this._element.classList.add(co,ho),this._queueCallback((()=>{this._element.classList.remove(ho),fe.trigger(this._element,ao),this._maybeScheduleHide()}),this._element,this._config.animation))}hide(){this.isShown()&&(fe.trigger(this._element,so).defaultPrevented||(this._element.classList.add(ho),this._queueCallback((()=>{this._element.classList.add(lo),this._element.classList.remove(ho,co),fe.trigger(this._element,oo)}),this._element,this._config.animation)))}dispose(){this._clearTimeout(),this.isShown()&&this._element.classList.remove(co),super.dispose()}isShown(){return this._element.classList.contains(co)}_maybeScheduleHide(){this._config.autohide&&(this._hasMouseInteraction||this._hasKeyboardInteraction||(this._timeout=setTimeout((()=>{this.hide()}),this._config.delay)))}_onInteraction(t,e){switch(t.type){case"mouseover":case"mouseout":this._hasMouseInteraction=e;break;case"focusin":case"focusout":this._hasKeyboardInteraction=e}if(e)return void this._clearTimeout();const i=t.relatedTarget;this._element===i||this._element.contains(i)||this._maybeScheduleHide()}_setListeners(){fe.on(this._element,to,(t=>this._onInteraction(t,!0))),fe.on(this._element,eo,(t=>this._onInteraction(t,!1))),fe.on(this._element,io,(t=>this._onInteraction(t,!0))),fe.on(this._element,no,(t=>this._onInteraction(t,!1)))}_clearTimeout(){clearTimeout(this._timeout),this._timeout=null}static jQueryInterface(t){return this.each((function(){const e=po.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}function mo(t){"loading"!=document.readyState?t():document.addEventListener("DOMContentLoaded",t)}Ee(po),Qt(po),mo((function(){[].slice.call(document.querySelectorAll('[data-bs-toggle="tooltip"]')).map((function(t){return new fs(t,{delay:{show:500,hide:100}})}))})),mo((function(){document.getElementById("pst-back-to-top").addEventListener("click",(function(){document.body.scrollTop=0,document.documentElement.scrollTop=0}))})),mo((function(){var t=document.getElementById("pst-back-to-top"),e=document.getElementsByClassName("bd-header")[0].getBoundingClientRect();window.addEventListener("scroll",(function(){this.oldScroll>this.scrollY&&this.scrollY>e.bottom?t.style.display="block":t.style.display="none",this.oldScroll=this.scrollY}))})),window.bootstrap=i})(); +//# sourceMappingURL=bootstrap.js.map \ No newline at end of file diff --git a/_static/scripts/bootstrap.js.LICENSE.txt b/_static/scripts/bootstrap.js.LICENSE.txt new file mode 100644 index 00000000..28755c2c --- /dev/null +++ b/_static/scripts/bootstrap.js.LICENSE.txt @@ -0,0 +1,5 @@ +/*! + * Bootstrap v5.3.3 (https://getbootstrap.com/) + * Copyright 2011-2024 The Bootstrap Authors (https://github.com/twbs/bootstrap/graphs/contributors) + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + */ diff --git a/_static/scripts/bootstrap.js.map b/_static/scripts/bootstrap.js.map new file mode 100644 index 00000000..e9e81589 --- /dev/null +++ b/_static/scripts/bootstrap.js.map @@ -0,0 +1 @@ +{"version":3,"file":"scripts/bootstrap.js","mappings":"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","sources":["webpack://pydata_sphinx_theme/webpack/bootstrap","webpack://pydata_sphinx_theme/webpack/runtime/define property getters","webpack://pydata_sphinx_theme/webpack/runtime/hasOwnProperty shorthand","webpack://pydata_sphinx_theme/webpack/runtime/make namespace object","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/enums.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getNodeName.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getWindow.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/instanceOf.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/applyStyles.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getBasePlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/math.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/userAgent.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/isLayoutViewport.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getBoundingClientRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getLayoutRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/contains.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getComputedStyle.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/isTableElement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getDocumentElement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getParentNode.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getOffsetParent.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getMainAxisFromPlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/within.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/mergePaddingObject.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getFreshSideObject.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/expandToHashMap.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/arrow.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getVariation.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/computeStyles.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/eventListeners.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getOppositePlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getOppositeVariationPlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getWindowScroll.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getWindowScrollBarX.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/isScrollParent.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getScrollParent.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/listScrollParents.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/rectToClientRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getClippingRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getViewportRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getDocumentRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/computeOffsets.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/detectOverflow.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/flip.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/computeAutoPlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/hide.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/offset.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/popperOffsets.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/preventOverflow.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getAltAxis.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getCompositeRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getNodeScroll.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getHTMLElementScroll.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/orderModifiers.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/createPopper.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/debounce.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/mergeByName.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/popper.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/popper-lite.js","webpack://pydata_sphinx_theme/./node_modules/bootstrap/dist/js/bootstrap.esm.js","webpack://pydata_sphinx_theme/./src/pydata_sphinx_theme/assets/scripts/mixin.js","webpack://pydata_sphinx_theme/./src/pydata_sphinx_theme/assets/scripts/bootstrap.js"],"sourcesContent":["// The require scope\nvar __webpack_require__ = {};\n\n","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","export var top = 'top';\nexport var bottom = 'bottom';\nexport var right = 'right';\nexport var left = 'left';\nexport var auto = 'auto';\nexport var basePlacements = [top, bottom, right, left];\nexport var start = 'start';\nexport var end = 'end';\nexport var clippingParents = 'clippingParents';\nexport var viewport = 'viewport';\nexport var popper = 'popper';\nexport var reference = 'reference';\nexport var variationPlacements = /*#__PURE__*/basePlacements.reduce(function (acc, placement) {\n return acc.concat([placement + \"-\" + start, placement + \"-\" + end]);\n}, []);\nexport var placements = /*#__PURE__*/[].concat(basePlacements, [auto]).reduce(function (acc, placement) {\n return acc.concat([placement, placement + \"-\" + start, placement + \"-\" + end]);\n}, []); // modifiers that need to read the DOM\n\nexport var beforeRead = 'beforeRead';\nexport var read = 'read';\nexport var afterRead = 'afterRead'; // pure-logic modifiers\n\nexport var beforeMain = 'beforeMain';\nexport var main = 'main';\nexport var afterMain = 'afterMain'; // modifier with the purpose to write to the DOM (or write into a framework state)\n\nexport var beforeWrite = 'beforeWrite';\nexport var write = 'write';\nexport var afterWrite = 'afterWrite';\nexport var modifierPhases = [beforeRead, read, afterRead, beforeMain, main, afterMain, beforeWrite, write, afterWrite];","export default function getNodeName(element) {\n return element ? (element.nodeName || '').toLowerCase() : null;\n}","export default function getWindow(node) {\n if (node == null) {\n return window;\n }\n\n if (node.toString() !== '[object Window]') {\n var ownerDocument = node.ownerDocument;\n return ownerDocument ? ownerDocument.defaultView || window : window;\n }\n\n return node;\n}","import getWindow from \"./getWindow.js\";\n\nfunction isElement(node) {\n var OwnElement = getWindow(node).Element;\n return node instanceof OwnElement || node instanceof Element;\n}\n\nfunction isHTMLElement(node) {\n var OwnElement = getWindow(node).HTMLElement;\n return node instanceof OwnElement || node instanceof HTMLElement;\n}\n\nfunction isShadowRoot(node) {\n // IE 11 has no ShadowRoot\n if (typeof ShadowRoot === 'undefined') {\n return false;\n }\n\n var OwnElement = getWindow(node).ShadowRoot;\n return node instanceof OwnElement || node instanceof ShadowRoot;\n}\n\nexport { isElement, isHTMLElement, isShadowRoot };","import getNodeName from \"../dom-utils/getNodeName.js\";\nimport { isHTMLElement } from \"../dom-utils/instanceOf.js\"; // This modifier takes the styles prepared by the `computeStyles` modifier\n// and applies them to the HTMLElements such as popper and arrow\n\nfunction applyStyles(_ref) {\n var state = _ref.state;\n Object.keys(state.elements).forEach(function (name) {\n var style = state.styles[name] || {};\n var attributes = state.attributes[name] || {};\n var element = state.elements[name]; // arrow is optional + virtual elements\n\n if (!isHTMLElement(element) || !getNodeName(element)) {\n return;\n } // Flow doesn't support to extend this property, but it's the most\n // effective way to apply styles to an HTMLElement\n // $FlowFixMe[cannot-write]\n\n\n Object.assign(element.style, style);\n Object.keys(attributes).forEach(function (name) {\n var value = attributes[name];\n\n if (value === false) {\n element.removeAttribute(name);\n } else {\n element.setAttribute(name, value === true ? '' : value);\n }\n });\n });\n}\n\nfunction effect(_ref2) {\n var state = _ref2.state;\n var initialStyles = {\n popper: {\n position: state.options.strategy,\n left: '0',\n top: '0',\n margin: '0'\n },\n arrow: {\n position: 'absolute'\n },\n reference: {}\n };\n Object.assign(state.elements.popper.style, initialStyles.popper);\n state.styles = initialStyles;\n\n if (state.elements.arrow) {\n Object.assign(state.elements.arrow.style, initialStyles.arrow);\n }\n\n return function () {\n Object.keys(state.elements).forEach(function (name) {\n var element = state.elements[name];\n var attributes = state.attributes[name] || {};\n var styleProperties = Object.keys(state.styles.hasOwnProperty(name) ? state.styles[name] : initialStyles[name]); // Set all values to an empty string to unset them\n\n var style = styleProperties.reduce(function (style, property) {\n style[property] = '';\n return style;\n }, {}); // arrow is optional + virtual elements\n\n if (!isHTMLElement(element) || !getNodeName(element)) {\n return;\n }\n\n Object.assign(element.style, style);\n Object.keys(attributes).forEach(function (attribute) {\n element.removeAttribute(attribute);\n });\n });\n };\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'applyStyles',\n enabled: true,\n phase: 'write',\n fn: applyStyles,\n effect: effect,\n requires: ['computeStyles']\n};","import { auto } from \"../enums.js\";\nexport default function getBasePlacement(placement) {\n return placement.split('-')[0];\n}","export var max = Math.max;\nexport var min = Math.min;\nexport var round = Math.round;","export default function getUAString() {\n var uaData = navigator.userAgentData;\n\n if (uaData != null && uaData.brands && Array.isArray(uaData.brands)) {\n return uaData.brands.map(function (item) {\n return item.brand + \"/\" + item.version;\n }).join(' ');\n }\n\n return navigator.userAgent;\n}","import getUAString from \"../utils/userAgent.js\";\nexport default function isLayoutViewport() {\n return !/^((?!chrome|android).)*safari/i.test(getUAString());\n}","import { isElement, isHTMLElement } from \"./instanceOf.js\";\nimport { round } from \"../utils/math.js\";\nimport getWindow from \"./getWindow.js\";\nimport isLayoutViewport from \"./isLayoutViewport.js\";\nexport default function getBoundingClientRect(element, includeScale, isFixedStrategy) {\n if (includeScale === void 0) {\n includeScale = false;\n }\n\n if (isFixedStrategy === void 0) {\n isFixedStrategy = false;\n }\n\n var clientRect = element.getBoundingClientRect();\n var scaleX = 1;\n var scaleY = 1;\n\n if (includeScale && isHTMLElement(element)) {\n scaleX = element.offsetWidth > 0 ? round(clientRect.width) / element.offsetWidth || 1 : 1;\n scaleY = element.offsetHeight > 0 ? round(clientRect.height) / element.offsetHeight || 1 : 1;\n }\n\n var _ref = isElement(element) ? getWindow(element) : window,\n visualViewport = _ref.visualViewport;\n\n var addVisualOffsets = !isLayoutViewport() && isFixedStrategy;\n var x = (clientRect.left + (addVisualOffsets && visualViewport ? visualViewport.offsetLeft : 0)) / scaleX;\n var y = (clientRect.top + (addVisualOffsets && visualViewport ? visualViewport.offsetTop : 0)) / scaleY;\n var width = clientRect.width / scaleX;\n var height = clientRect.height / scaleY;\n return {\n width: width,\n height: height,\n top: y,\n right: x + width,\n bottom: y + height,\n left: x,\n x: x,\n y: y\n };\n}","import getBoundingClientRect from \"./getBoundingClientRect.js\"; // Returns the layout rect of an element relative to its offsetParent. Layout\n// means it doesn't take into account transforms.\n\nexport default function getLayoutRect(element) {\n var clientRect = getBoundingClientRect(element); // Use the clientRect sizes if it's not been transformed.\n // Fixes https://github.com/popperjs/popper-core/issues/1223\n\n var width = element.offsetWidth;\n var height = element.offsetHeight;\n\n if (Math.abs(clientRect.width - width) <= 1) {\n width = clientRect.width;\n }\n\n if (Math.abs(clientRect.height - height) <= 1) {\n height = clientRect.height;\n }\n\n return {\n x: element.offsetLeft,\n y: element.offsetTop,\n width: width,\n height: height\n };\n}","import { isShadowRoot } from \"./instanceOf.js\";\nexport default function contains(parent, child) {\n var rootNode = child.getRootNode && child.getRootNode(); // First, attempt with faster native method\n\n if (parent.contains(child)) {\n return true;\n } // then fallback to custom implementation with Shadow DOM support\n else if (rootNode && isShadowRoot(rootNode)) {\n var next = child;\n\n do {\n if (next && parent.isSameNode(next)) {\n return true;\n } // $FlowFixMe[prop-missing]: need a better way to handle this...\n\n\n next = next.parentNode || next.host;\n } while (next);\n } // Give up, the result is false\n\n\n return false;\n}","import getWindow from \"./getWindow.js\";\nexport default function getComputedStyle(element) {\n return getWindow(element).getComputedStyle(element);\n}","import getNodeName from \"./getNodeName.js\";\nexport default function isTableElement(element) {\n return ['table', 'td', 'th'].indexOf(getNodeName(element)) >= 0;\n}","import { isElement } from \"./instanceOf.js\";\nexport default function getDocumentElement(element) {\n // $FlowFixMe[incompatible-return]: assume body is always available\n return ((isElement(element) ? element.ownerDocument : // $FlowFixMe[prop-missing]\n element.document) || window.document).documentElement;\n}","import getNodeName from \"./getNodeName.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport { isShadowRoot } from \"./instanceOf.js\";\nexport default function getParentNode(element) {\n if (getNodeName(element) === 'html') {\n return element;\n }\n\n return (// this is a quicker (but less type safe) way to save quite some bytes from the bundle\n // $FlowFixMe[incompatible-return]\n // $FlowFixMe[prop-missing]\n element.assignedSlot || // step into the shadow DOM of the parent of a slotted node\n element.parentNode || ( // DOM Element detected\n isShadowRoot(element) ? element.host : null) || // ShadowRoot detected\n // $FlowFixMe[incompatible-call]: HTMLElement is a Node\n getDocumentElement(element) // fallback\n\n );\n}","import getWindow from \"./getWindow.js\";\nimport getNodeName from \"./getNodeName.js\";\nimport getComputedStyle from \"./getComputedStyle.js\";\nimport { isHTMLElement, isShadowRoot } from \"./instanceOf.js\";\nimport isTableElement from \"./isTableElement.js\";\nimport getParentNode from \"./getParentNode.js\";\nimport getUAString from \"../utils/userAgent.js\";\n\nfunction getTrueOffsetParent(element) {\n if (!isHTMLElement(element) || // https://github.com/popperjs/popper-core/issues/837\n getComputedStyle(element).position === 'fixed') {\n return null;\n }\n\n return element.offsetParent;\n} // `.offsetParent` reports `null` for fixed elements, while absolute elements\n// return the containing block\n\n\nfunction getContainingBlock(element) {\n var isFirefox = /firefox/i.test(getUAString());\n var isIE = /Trident/i.test(getUAString());\n\n if (isIE && isHTMLElement(element)) {\n // In IE 9, 10 and 11 fixed elements containing block is always established by the viewport\n var elementCss = getComputedStyle(element);\n\n if (elementCss.position === 'fixed') {\n return null;\n }\n }\n\n var currentNode = getParentNode(element);\n\n if (isShadowRoot(currentNode)) {\n currentNode = currentNode.host;\n }\n\n while (isHTMLElement(currentNode) && ['html', 'body'].indexOf(getNodeName(currentNode)) < 0) {\n var css = getComputedStyle(currentNode); // This is non-exhaustive but covers the most common CSS properties that\n // create a containing block.\n // https://developer.mozilla.org/en-US/docs/Web/CSS/Containing_block#identifying_the_containing_block\n\n if (css.transform !== 'none' || css.perspective !== 'none' || css.contain === 'paint' || ['transform', 'perspective'].indexOf(css.willChange) !== -1 || isFirefox && css.willChange === 'filter' || isFirefox && css.filter && css.filter !== 'none') {\n return currentNode;\n } else {\n currentNode = currentNode.parentNode;\n }\n }\n\n return null;\n} // Gets the closest ancestor positioned element. Handles some edge cases,\n// such as table ancestors and cross browser bugs.\n\n\nexport default function getOffsetParent(element) {\n var window = getWindow(element);\n var offsetParent = getTrueOffsetParent(element);\n\n while (offsetParent && isTableElement(offsetParent) && getComputedStyle(offsetParent).position === 'static') {\n offsetParent = getTrueOffsetParent(offsetParent);\n }\n\n if (offsetParent && (getNodeName(offsetParent) === 'html' || getNodeName(offsetParent) === 'body' && getComputedStyle(offsetParent).position === 'static')) {\n return window;\n }\n\n return offsetParent || getContainingBlock(element) || window;\n}","export default function getMainAxisFromPlacement(placement) {\n return ['top', 'bottom'].indexOf(placement) >= 0 ? 'x' : 'y';\n}","import { max as mathMax, min as mathMin } from \"./math.js\";\nexport function within(min, value, max) {\n return mathMax(min, mathMin(value, max));\n}\nexport function withinMaxClamp(min, value, max) {\n var v = within(min, value, max);\n return v > max ? max : v;\n}","import getFreshSideObject from \"./getFreshSideObject.js\";\nexport default function mergePaddingObject(paddingObject) {\n return Object.assign({}, getFreshSideObject(), paddingObject);\n}","export default function getFreshSideObject() {\n return {\n top: 0,\n right: 0,\n bottom: 0,\n left: 0\n };\n}","export default function expandToHashMap(value, keys) {\n return keys.reduce(function (hashMap, key) {\n hashMap[key] = value;\n return hashMap;\n }, {});\n}","import getBasePlacement from \"../utils/getBasePlacement.js\";\nimport getLayoutRect from \"../dom-utils/getLayoutRect.js\";\nimport contains from \"../dom-utils/contains.js\";\nimport getOffsetParent from \"../dom-utils/getOffsetParent.js\";\nimport getMainAxisFromPlacement from \"../utils/getMainAxisFromPlacement.js\";\nimport { within } from \"../utils/within.js\";\nimport mergePaddingObject from \"../utils/mergePaddingObject.js\";\nimport expandToHashMap from \"../utils/expandToHashMap.js\";\nimport { left, right, basePlacements, top, bottom } from \"../enums.js\"; // eslint-disable-next-line import/no-unused-modules\n\nvar toPaddingObject = function toPaddingObject(padding, state) {\n padding = typeof padding === 'function' ? padding(Object.assign({}, state.rects, {\n placement: state.placement\n })) : padding;\n return mergePaddingObject(typeof padding !== 'number' ? padding : expandToHashMap(padding, basePlacements));\n};\n\nfunction arrow(_ref) {\n var _state$modifiersData$;\n\n var state = _ref.state,\n name = _ref.name,\n options = _ref.options;\n var arrowElement = state.elements.arrow;\n var popperOffsets = state.modifiersData.popperOffsets;\n var basePlacement = getBasePlacement(state.placement);\n var axis = getMainAxisFromPlacement(basePlacement);\n var isVertical = [left, right].indexOf(basePlacement) >= 0;\n var len = isVertical ? 'height' : 'width';\n\n if (!arrowElement || !popperOffsets) {\n return;\n }\n\n var paddingObject = toPaddingObject(options.padding, state);\n var arrowRect = getLayoutRect(arrowElement);\n var minProp = axis === 'y' ? top : left;\n var maxProp = axis === 'y' ? bottom : right;\n var endDiff = state.rects.reference[len] + state.rects.reference[axis] - popperOffsets[axis] - state.rects.popper[len];\n var startDiff = popperOffsets[axis] - state.rects.reference[axis];\n var arrowOffsetParent = getOffsetParent(arrowElement);\n var clientSize = arrowOffsetParent ? axis === 'y' ? arrowOffsetParent.clientHeight || 0 : arrowOffsetParent.clientWidth || 0 : 0;\n var centerToReference = endDiff / 2 - startDiff / 2; // Make sure the arrow doesn't overflow the popper if the center point is\n // outside of the popper bounds\n\n var min = paddingObject[minProp];\n var max = clientSize - arrowRect[len] - paddingObject[maxProp];\n var center = clientSize / 2 - arrowRect[len] / 2 + centerToReference;\n var offset = within(min, center, max); // Prevents breaking syntax highlighting...\n\n var axisProp = axis;\n state.modifiersData[name] = (_state$modifiersData$ = {}, _state$modifiersData$[axisProp] = offset, _state$modifiersData$.centerOffset = offset - center, _state$modifiersData$);\n}\n\nfunction effect(_ref2) {\n var state = _ref2.state,\n options = _ref2.options;\n var _options$element = options.element,\n arrowElement = _options$element === void 0 ? '[data-popper-arrow]' : _options$element;\n\n if (arrowElement == null) {\n return;\n } // CSS selector\n\n\n if (typeof arrowElement === 'string') {\n arrowElement = state.elements.popper.querySelector(arrowElement);\n\n if (!arrowElement) {\n return;\n }\n }\n\n if (!contains(state.elements.popper, arrowElement)) {\n return;\n }\n\n state.elements.arrow = arrowElement;\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'arrow',\n enabled: true,\n phase: 'main',\n fn: arrow,\n effect: effect,\n requires: ['popperOffsets'],\n requiresIfExists: ['preventOverflow']\n};","export default function getVariation(placement) {\n return placement.split('-')[1];\n}","import { top, left, right, bottom, end } from \"../enums.js\";\nimport getOffsetParent from \"../dom-utils/getOffsetParent.js\";\nimport getWindow from \"../dom-utils/getWindow.js\";\nimport getDocumentElement from \"../dom-utils/getDocumentElement.js\";\nimport getComputedStyle from \"../dom-utils/getComputedStyle.js\";\nimport getBasePlacement from \"../utils/getBasePlacement.js\";\nimport getVariation from \"../utils/getVariation.js\";\nimport { round } from \"../utils/math.js\"; // eslint-disable-next-line import/no-unused-modules\n\nvar unsetSides = {\n top: 'auto',\n right: 'auto',\n bottom: 'auto',\n left: 'auto'\n}; // Round the offsets to the nearest suitable subpixel based on the DPR.\n// Zooming can change the DPR, but it seems to report a value that will\n// cleanly divide the values into the appropriate subpixels.\n\nfunction roundOffsetsByDPR(_ref, win) {\n var x = _ref.x,\n y = _ref.y;\n var dpr = win.devicePixelRatio || 1;\n return {\n x: round(x * dpr) / dpr || 0,\n y: round(y * dpr) / dpr || 0\n };\n}\n\nexport function mapToStyles(_ref2) {\n var _Object$assign2;\n\n var popper = _ref2.popper,\n popperRect = _ref2.popperRect,\n placement = _ref2.placement,\n variation = _ref2.variation,\n offsets = _ref2.offsets,\n position = _ref2.position,\n gpuAcceleration = _ref2.gpuAcceleration,\n adaptive = _ref2.adaptive,\n roundOffsets = _ref2.roundOffsets,\n isFixed = _ref2.isFixed;\n var _offsets$x = offsets.x,\n x = _offsets$x === void 0 ? 0 : _offsets$x,\n _offsets$y = offsets.y,\n y = _offsets$y === void 0 ? 0 : _offsets$y;\n\n var _ref3 = typeof roundOffsets === 'function' ? roundOffsets({\n x: x,\n y: y\n }) : {\n x: x,\n y: y\n };\n\n x = _ref3.x;\n y = _ref3.y;\n var hasX = offsets.hasOwnProperty('x');\n var hasY = offsets.hasOwnProperty('y');\n var sideX = left;\n var sideY = top;\n var win = window;\n\n if (adaptive) {\n var offsetParent = getOffsetParent(popper);\n var heightProp = 'clientHeight';\n var widthProp = 'clientWidth';\n\n if (offsetParent === getWindow(popper)) {\n offsetParent = getDocumentElement(popper);\n\n if (getComputedStyle(offsetParent).position !== 'static' && position === 'absolute') {\n heightProp = 'scrollHeight';\n widthProp = 'scrollWidth';\n }\n } // $FlowFixMe[incompatible-cast]: force type refinement, we compare offsetParent with window above, but Flow doesn't detect it\n\n\n offsetParent = offsetParent;\n\n if (placement === top || (placement === left || placement === right) && variation === end) {\n sideY = bottom;\n var offsetY = isFixed && offsetParent === win && win.visualViewport ? win.visualViewport.height : // $FlowFixMe[prop-missing]\n offsetParent[heightProp];\n y -= offsetY - popperRect.height;\n y *= gpuAcceleration ? 1 : -1;\n }\n\n if (placement === left || (placement === top || placement === bottom) && variation === end) {\n sideX = right;\n var offsetX = isFixed && offsetParent === win && win.visualViewport ? win.visualViewport.width : // $FlowFixMe[prop-missing]\n offsetParent[widthProp];\n x -= offsetX - popperRect.width;\n x *= gpuAcceleration ? 1 : -1;\n }\n }\n\n var commonStyles = Object.assign({\n position: position\n }, adaptive && unsetSides);\n\n var _ref4 = roundOffsets === true ? roundOffsetsByDPR({\n x: x,\n y: y\n }, getWindow(popper)) : {\n x: x,\n y: y\n };\n\n x = _ref4.x;\n y = _ref4.y;\n\n if (gpuAcceleration) {\n var _Object$assign;\n\n return Object.assign({}, commonStyles, (_Object$assign = {}, _Object$assign[sideY] = hasY ? '0' : '', _Object$assign[sideX] = hasX ? '0' : '', _Object$assign.transform = (win.devicePixelRatio || 1) <= 1 ? \"translate(\" + x + \"px, \" + y + \"px)\" : \"translate3d(\" + x + \"px, \" + y + \"px, 0)\", _Object$assign));\n }\n\n return Object.assign({}, commonStyles, (_Object$assign2 = {}, _Object$assign2[sideY] = hasY ? y + \"px\" : '', _Object$assign2[sideX] = hasX ? x + \"px\" : '', _Object$assign2.transform = '', _Object$assign2));\n}\n\nfunction computeStyles(_ref5) {\n var state = _ref5.state,\n options = _ref5.options;\n var _options$gpuAccelerat = options.gpuAcceleration,\n gpuAcceleration = _options$gpuAccelerat === void 0 ? true : _options$gpuAccelerat,\n _options$adaptive = options.adaptive,\n adaptive = _options$adaptive === void 0 ? true : _options$adaptive,\n _options$roundOffsets = options.roundOffsets,\n roundOffsets = _options$roundOffsets === void 0 ? true : _options$roundOffsets;\n var commonStyles = {\n placement: getBasePlacement(state.placement),\n variation: getVariation(state.placement),\n popper: state.elements.popper,\n popperRect: state.rects.popper,\n gpuAcceleration: gpuAcceleration,\n isFixed: state.options.strategy === 'fixed'\n };\n\n if (state.modifiersData.popperOffsets != null) {\n state.styles.popper = Object.assign({}, state.styles.popper, mapToStyles(Object.assign({}, commonStyles, {\n offsets: state.modifiersData.popperOffsets,\n position: state.options.strategy,\n adaptive: adaptive,\n roundOffsets: roundOffsets\n })));\n }\n\n if (state.modifiersData.arrow != null) {\n state.styles.arrow = Object.assign({}, state.styles.arrow, mapToStyles(Object.assign({}, commonStyles, {\n offsets: state.modifiersData.arrow,\n position: 'absolute',\n adaptive: false,\n roundOffsets: roundOffsets\n })));\n }\n\n state.attributes.popper = Object.assign({}, state.attributes.popper, {\n 'data-popper-placement': state.placement\n });\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'computeStyles',\n enabled: true,\n phase: 'beforeWrite',\n fn: computeStyles,\n data: {}\n};","import getWindow from \"../dom-utils/getWindow.js\"; // eslint-disable-next-line import/no-unused-modules\n\nvar passive = {\n passive: true\n};\n\nfunction effect(_ref) {\n var state = _ref.state,\n instance = _ref.instance,\n options = _ref.options;\n var _options$scroll = options.scroll,\n scroll = _options$scroll === void 0 ? true : _options$scroll,\n _options$resize = options.resize,\n resize = _options$resize === void 0 ? true : _options$resize;\n var window = getWindow(state.elements.popper);\n var scrollParents = [].concat(state.scrollParents.reference, state.scrollParents.popper);\n\n if (scroll) {\n scrollParents.forEach(function (scrollParent) {\n scrollParent.addEventListener('scroll', instance.update, passive);\n });\n }\n\n if (resize) {\n window.addEventListener('resize', instance.update, passive);\n }\n\n return function () {\n if (scroll) {\n scrollParents.forEach(function (scrollParent) {\n scrollParent.removeEventListener('scroll', instance.update, passive);\n });\n }\n\n if (resize) {\n window.removeEventListener('resize', instance.update, passive);\n }\n };\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'eventListeners',\n enabled: true,\n phase: 'write',\n fn: function fn() {},\n effect: effect,\n data: {}\n};","var hash = {\n left: 'right',\n right: 'left',\n bottom: 'top',\n top: 'bottom'\n};\nexport default function getOppositePlacement(placement) {\n return placement.replace(/left|right|bottom|top/g, function (matched) {\n return hash[matched];\n });\n}","var hash = {\n start: 'end',\n end: 'start'\n};\nexport default function getOppositeVariationPlacement(placement) {\n return placement.replace(/start|end/g, function (matched) {\n return hash[matched];\n });\n}","import getWindow from \"./getWindow.js\";\nexport default function getWindowScroll(node) {\n var win = getWindow(node);\n var scrollLeft = win.pageXOffset;\n var scrollTop = win.pageYOffset;\n return {\n scrollLeft: scrollLeft,\n scrollTop: scrollTop\n };\n}","import getBoundingClientRect from \"./getBoundingClientRect.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport getWindowScroll from \"./getWindowScroll.js\";\nexport default function getWindowScrollBarX(element) {\n // If has a CSS width greater than the viewport, then this will be\n // incorrect for RTL.\n // Popper 1 is broken in this case and never had a bug report so let's assume\n // it's not an issue. I don't think anyone ever specifies width on \n // anyway.\n // Browsers where the left scrollbar doesn't cause an issue report `0` for\n // this (e.g. Edge 2019, IE11, Safari)\n return getBoundingClientRect(getDocumentElement(element)).left + getWindowScroll(element).scrollLeft;\n}","import getComputedStyle from \"./getComputedStyle.js\";\nexport default function isScrollParent(element) {\n // Firefox wants us to check `-x` and `-y` variations as well\n var _getComputedStyle = getComputedStyle(element),\n overflow = _getComputedStyle.overflow,\n overflowX = _getComputedStyle.overflowX,\n overflowY = _getComputedStyle.overflowY;\n\n return /auto|scroll|overlay|hidden/.test(overflow + overflowY + overflowX);\n}","import getParentNode from \"./getParentNode.js\";\nimport isScrollParent from \"./isScrollParent.js\";\nimport getNodeName from \"./getNodeName.js\";\nimport { isHTMLElement } from \"./instanceOf.js\";\nexport default function getScrollParent(node) {\n if (['html', 'body', '#document'].indexOf(getNodeName(node)) >= 0) {\n // $FlowFixMe[incompatible-return]: assume body is always available\n return node.ownerDocument.body;\n }\n\n if (isHTMLElement(node) && isScrollParent(node)) {\n return node;\n }\n\n return getScrollParent(getParentNode(node));\n}","import getScrollParent from \"./getScrollParent.js\";\nimport getParentNode from \"./getParentNode.js\";\nimport getWindow from \"./getWindow.js\";\nimport isScrollParent from \"./isScrollParent.js\";\n/*\ngiven a DOM element, return the list of all scroll parents, up the list of ancesors\nuntil we get to the top window object. This list is what we attach scroll listeners\nto, because if any of these parent elements scroll, we'll need to re-calculate the\nreference element's position.\n*/\n\nexport default function listScrollParents(element, list) {\n var _element$ownerDocumen;\n\n if (list === void 0) {\n list = [];\n }\n\n var scrollParent = getScrollParent(element);\n var isBody = scrollParent === ((_element$ownerDocumen = element.ownerDocument) == null ? void 0 : _element$ownerDocumen.body);\n var win = getWindow(scrollParent);\n var target = isBody ? [win].concat(win.visualViewport || [], isScrollParent(scrollParent) ? scrollParent : []) : scrollParent;\n var updatedList = list.concat(target);\n return isBody ? updatedList : // $FlowFixMe[incompatible-call]: isBody tells us target will be an HTMLElement here\n updatedList.concat(listScrollParents(getParentNode(target)));\n}","export default function rectToClientRect(rect) {\n return Object.assign({}, rect, {\n left: rect.x,\n top: rect.y,\n right: rect.x + rect.width,\n bottom: rect.y + rect.height\n });\n}","import { viewport } from \"../enums.js\";\nimport getViewportRect from \"./getViewportRect.js\";\nimport getDocumentRect from \"./getDocumentRect.js\";\nimport listScrollParents from \"./listScrollParents.js\";\nimport getOffsetParent from \"./getOffsetParent.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport getComputedStyle from \"./getComputedStyle.js\";\nimport { isElement, isHTMLElement } from \"./instanceOf.js\";\nimport getBoundingClientRect from \"./getBoundingClientRect.js\";\nimport getParentNode from \"./getParentNode.js\";\nimport contains from \"./contains.js\";\nimport getNodeName from \"./getNodeName.js\";\nimport rectToClientRect from \"../utils/rectToClientRect.js\";\nimport { max, min } from \"../utils/math.js\";\n\nfunction getInnerBoundingClientRect(element, strategy) {\n var rect = getBoundingClientRect(element, false, strategy === 'fixed');\n rect.top = rect.top + element.clientTop;\n rect.left = rect.left + element.clientLeft;\n rect.bottom = rect.top + element.clientHeight;\n rect.right = rect.left + element.clientWidth;\n rect.width = element.clientWidth;\n rect.height = element.clientHeight;\n rect.x = rect.left;\n rect.y = rect.top;\n return rect;\n}\n\nfunction getClientRectFromMixedType(element, clippingParent, strategy) {\n return clippingParent === viewport ? rectToClientRect(getViewportRect(element, strategy)) : isElement(clippingParent) ? getInnerBoundingClientRect(clippingParent, strategy) : rectToClientRect(getDocumentRect(getDocumentElement(element)));\n} // A \"clipping parent\" is an overflowable container with the characteristic of\n// clipping (or hiding) overflowing elements with a position different from\n// `initial`\n\n\nfunction getClippingParents(element) {\n var clippingParents = listScrollParents(getParentNode(element));\n var canEscapeClipping = ['absolute', 'fixed'].indexOf(getComputedStyle(element).position) >= 0;\n var clipperElement = canEscapeClipping && isHTMLElement(element) ? getOffsetParent(element) : element;\n\n if (!isElement(clipperElement)) {\n return [];\n } // $FlowFixMe[incompatible-return]: https://github.com/facebook/flow/issues/1414\n\n\n return clippingParents.filter(function (clippingParent) {\n return isElement(clippingParent) && contains(clippingParent, clipperElement) && getNodeName(clippingParent) !== 'body';\n });\n} // Gets the maximum area that the element is visible in due to any number of\n// clipping parents\n\n\nexport default function getClippingRect(element, boundary, rootBoundary, strategy) {\n var mainClippingParents = boundary === 'clippingParents' ? getClippingParents(element) : [].concat(boundary);\n var clippingParents = [].concat(mainClippingParents, [rootBoundary]);\n var firstClippingParent = clippingParents[0];\n var clippingRect = clippingParents.reduce(function (accRect, clippingParent) {\n var rect = getClientRectFromMixedType(element, clippingParent, strategy);\n accRect.top = max(rect.top, accRect.top);\n accRect.right = min(rect.right, accRect.right);\n accRect.bottom = min(rect.bottom, accRect.bottom);\n accRect.left = max(rect.left, accRect.left);\n return accRect;\n }, getClientRectFromMixedType(element, firstClippingParent, strategy));\n clippingRect.width = clippingRect.right - clippingRect.left;\n clippingRect.height = clippingRect.bottom - clippingRect.top;\n clippingRect.x = clippingRect.left;\n clippingRect.y = clippingRect.top;\n return clippingRect;\n}","import getWindow from \"./getWindow.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport getWindowScrollBarX from \"./getWindowScrollBarX.js\";\nimport isLayoutViewport from \"./isLayoutViewport.js\";\nexport default function getViewportRect(element, strategy) {\n var win = getWindow(element);\n var html = getDocumentElement(element);\n var visualViewport = win.visualViewport;\n var width = html.clientWidth;\n var height = html.clientHeight;\n var x = 0;\n var y = 0;\n\n if (visualViewport) {\n width = visualViewport.width;\n height = visualViewport.height;\n var layoutViewport = isLayoutViewport();\n\n if (layoutViewport || !layoutViewport && strategy === 'fixed') {\n x = visualViewport.offsetLeft;\n y = visualViewport.offsetTop;\n }\n }\n\n return {\n width: width,\n height: height,\n x: x + getWindowScrollBarX(element),\n y: y\n };\n}","import getDocumentElement from \"./getDocumentElement.js\";\nimport getComputedStyle from \"./getComputedStyle.js\";\nimport getWindowScrollBarX from \"./getWindowScrollBarX.js\";\nimport getWindowScroll from \"./getWindowScroll.js\";\nimport { max } from \"../utils/math.js\"; // Gets the entire size of the scrollable document area, even extending outside\n// of the `` and `` rect bounds if horizontally scrollable\n\nexport default function getDocumentRect(element) {\n var _element$ownerDocumen;\n\n var html = getDocumentElement(element);\n var winScroll = getWindowScroll(element);\n var body = (_element$ownerDocumen = element.ownerDocument) == null ? void 0 : _element$ownerDocumen.body;\n var width = max(html.scrollWidth, html.clientWidth, body ? body.scrollWidth : 0, body ? body.clientWidth : 0);\n var height = max(html.scrollHeight, html.clientHeight, body ? body.scrollHeight : 0, body ? body.clientHeight : 0);\n var x = -winScroll.scrollLeft + getWindowScrollBarX(element);\n var y = -winScroll.scrollTop;\n\n if (getComputedStyle(body || html).direction === 'rtl') {\n x += max(html.clientWidth, body ? body.clientWidth : 0) - width;\n }\n\n return {\n width: width,\n height: height,\n x: x,\n y: y\n };\n}","import getBasePlacement from \"./getBasePlacement.js\";\nimport getVariation from \"./getVariation.js\";\nimport getMainAxisFromPlacement from \"./getMainAxisFromPlacement.js\";\nimport { top, right, bottom, left, start, end } from \"../enums.js\";\nexport default function computeOffsets(_ref) {\n var reference = _ref.reference,\n element = _ref.element,\n placement = _ref.placement;\n var basePlacement = placement ? getBasePlacement(placement) : null;\n var variation = placement ? getVariation(placement) : null;\n var commonX = reference.x + reference.width / 2 - element.width / 2;\n var commonY = reference.y + reference.height / 2 - element.height / 2;\n var offsets;\n\n switch (basePlacement) {\n case top:\n offsets = {\n x: commonX,\n y: reference.y - element.height\n };\n break;\n\n case bottom:\n offsets = {\n x: commonX,\n y: reference.y + reference.height\n };\n break;\n\n case right:\n offsets = {\n x: reference.x + reference.width,\n y: commonY\n };\n break;\n\n case left:\n offsets = {\n x: reference.x - element.width,\n y: commonY\n };\n break;\n\n default:\n offsets = {\n x: reference.x,\n y: reference.y\n };\n }\n\n var mainAxis = basePlacement ? getMainAxisFromPlacement(basePlacement) : null;\n\n if (mainAxis != null) {\n var len = mainAxis === 'y' ? 'height' : 'width';\n\n switch (variation) {\n case start:\n offsets[mainAxis] = offsets[mainAxis] - (reference[len] / 2 - element[len] / 2);\n break;\n\n case end:\n offsets[mainAxis] = offsets[mainAxis] + (reference[len] / 2 - element[len] / 2);\n break;\n\n default:\n }\n }\n\n return offsets;\n}","import getClippingRect from \"../dom-utils/getClippingRect.js\";\nimport getDocumentElement from \"../dom-utils/getDocumentElement.js\";\nimport getBoundingClientRect from \"../dom-utils/getBoundingClientRect.js\";\nimport computeOffsets from \"./computeOffsets.js\";\nimport rectToClientRect from \"./rectToClientRect.js\";\nimport { clippingParents, reference, popper, bottom, top, right, basePlacements, viewport } from \"../enums.js\";\nimport { isElement } from \"../dom-utils/instanceOf.js\";\nimport mergePaddingObject from \"./mergePaddingObject.js\";\nimport expandToHashMap from \"./expandToHashMap.js\"; // eslint-disable-next-line import/no-unused-modules\n\nexport default function detectOverflow(state, options) {\n if (options === void 0) {\n options = {};\n }\n\n var _options = options,\n _options$placement = _options.placement,\n placement = _options$placement === void 0 ? state.placement : _options$placement,\n _options$strategy = _options.strategy,\n strategy = _options$strategy === void 0 ? state.strategy : _options$strategy,\n _options$boundary = _options.boundary,\n boundary = _options$boundary === void 0 ? clippingParents : _options$boundary,\n _options$rootBoundary = _options.rootBoundary,\n rootBoundary = _options$rootBoundary === void 0 ? viewport : _options$rootBoundary,\n _options$elementConte = _options.elementContext,\n elementContext = _options$elementConte === void 0 ? popper : _options$elementConte,\n _options$altBoundary = _options.altBoundary,\n altBoundary = _options$altBoundary === void 0 ? false : _options$altBoundary,\n _options$padding = _options.padding,\n padding = _options$padding === void 0 ? 0 : _options$padding;\n var paddingObject = mergePaddingObject(typeof padding !== 'number' ? padding : expandToHashMap(padding, basePlacements));\n var altContext = elementContext === popper ? reference : popper;\n var popperRect = state.rects.popper;\n var element = state.elements[altBoundary ? altContext : elementContext];\n var clippingClientRect = getClippingRect(isElement(element) ? element : element.contextElement || getDocumentElement(state.elements.popper), boundary, rootBoundary, strategy);\n var referenceClientRect = getBoundingClientRect(state.elements.reference);\n var popperOffsets = computeOffsets({\n reference: referenceClientRect,\n element: popperRect,\n strategy: 'absolute',\n placement: placement\n });\n var popperClientRect = rectToClientRect(Object.assign({}, popperRect, popperOffsets));\n var elementClientRect = elementContext === popper ? popperClientRect : referenceClientRect; // positive = overflowing the clipping rect\n // 0 or negative = within the clipping rect\n\n var overflowOffsets = {\n top: clippingClientRect.top - elementClientRect.top + paddingObject.top,\n bottom: elementClientRect.bottom - clippingClientRect.bottom + paddingObject.bottom,\n left: clippingClientRect.left - elementClientRect.left + paddingObject.left,\n right: elementClientRect.right - clippingClientRect.right + paddingObject.right\n };\n var offsetData = state.modifiersData.offset; // Offsets can be applied only to the popper element\n\n if (elementContext === popper && offsetData) {\n var offset = offsetData[placement];\n Object.keys(overflowOffsets).forEach(function (key) {\n var multiply = [right, bottom].indexOf(key) >= 0 ? 1 : -1;\n var axis = [top, bottom].indexOf(key) >= 0 ? 'y' : 'x';\n overflowOffsets[key] += offset[axis] * multiply;\n });\n }\n\n return overflowOffsets;\n}","import getOppositePlacement from \"../utils/getOppositePlacement.js\";\nimport getBasePlacement from \"../utils/getBasePlacement.js\";\nimport getOppositeVariationPlacement from \"../utils/getOppositeVariationPlacement.js\";\nimport detectOverflow from \"../utils/detectOverflow.js\";\nimport computeAutoPlacement from \"../utils/computeAutoPlacement.js\";\nimport { bottom, top, start, right, left, auto } from \"../enums.js\";\nimport getVariation from \"../utils/getVariation.js\"; // eslint-disable-next-line import/no-unused-modules\n\nfunction getExpandedFallbackPlacements(placement) {\n if (getBasePlacement(placement) === auto) {\n return [];\n }\n\n var oppositePlacement = getOppositePlacement(placement);\n return [getOppositeVariationPlacement(placement), oppositePlacement, getOppositeVariationPlacement(oppositePlacement)];\n}\n\nfunction flip(_ref) {\n var state = _ref.state,\n options = _ref.options,\n name = _ref.name;\n\n if (state.modifiersData[name]._skip) {\n return;\n }\n\n var _options$mainAxis = options.mainAxis,\n checkMainAxis = _options$mainAxis === void 0 ? true : _options$mainAxis,\n _options$altAxis = options.altAxis,\n checkAltAxis = _options$altAxis === void 0 ? true : _options$altAxis,\n specifiedFallbackPlacements = options.fallbackPlacements,\n padding = options.padding,\n boundary = options.boundary,\n rootBoundary = options.rootBoundary,\n altBoundary = options.altBoundary,\n _options$flipVariatio = options.flipVariations,\n flipVariations = _options$flipVariatio === void 0 ? true : _options$flipVariatio,\n allowedAutoPlacements = options.allowedAutoPlacements;\n var preferredPlacement = state.options.placement;\n var basePlacement = getBasePlacement(preferredPlacement);\n var isBasePlacement = basePlacement === preferredPlacement;\n var fallbackPlacements = specifiedFallbackPlacements || (isBasePlacement || !flipVariations ? [getOppositePlacement(preferredPlacement)] : getExpandedFallbackPlacements(preferredPlacement));\n var placements = [preferredPlacement].concat(fallbackPlacements).reduce(function (acc, placement) {\n return acc.concat(getBasePlacement(placement) === auto ? computeAutoPlacement(state, {\n placement: placement,\n boundary: boundary,\n rootBoundary: rootBoundary,\n padding: padding,\n flipVariations: flipVariations,\n allowedAutoPlacements: allowedAutoPlacements\n }) : placement);\n }, []);\n var referenceRect = state.rects.reference;\n var popperRect = state.rects.popper;\n var checksMap = new Map();\n var makeFallbackChecks = true;\n var firstFittingPlacement = placements[0];\n\n for (var i = 0; i < placements.length; i++) {\n var placement = placements[i];\n\n var _basePlacement = getBasePlacement(placement);\n\n var isStartVariation = getVariation(placement) === start;\n var isVertical = [top, bottom].indexOf(_basePlacement) >= 0;\n var len = isVertical ? 'width' : 'height';\n var overflow = detectOverflow(state, {\n placement: placement,\n boundary: boundary,\n rootBoundary: rootBoundary,\n altBoundary: altBoundary,\n padding: padding\n });\n var mainVariationSide = isVertical ? isStartVariation ? right : left : isStartVariation ? bottom : top;\n\n if (referenceRect[len] > popperRect[len]) {\n mainVariationSide = getOppositePlacement(mainVariationSide);\n }\n\n var altVariationSide = getOppositePlacement(mainVariationSide);\n var checks = [];\n\n if (checkMainAxis) {\n checks.push(overflow[_basePlacement] <= 0);\n }\n\n if (checkAltAxis) {\n checks.push(overflow[mainVariationSide] <= 0, overflow[altVariationSide] <= 0);\n }\n\n if (checks.every(function (check) {\n return check;\n })) {\n firstFittingPlacement = placement;\n makeFallbackChecks = false;\n break;\n }\n\n checksMap.set(placement, checks);\n }\n\n if (makeFallbackChecks) {\n // `2` may be desired in some cases – research later\n var numberOfChecks = flipVariations ? 3 : 1;\n\n var _loop = function _loop(_i) {\n var fittingPlacement = placements.find(function (placement) {\n var checks = checksMap.get(placement);\n\n if (checks) {\n return checks.slice(0, _i).every(function (check) {\n return check;\n });\n }\n });\n\n if (fittingPlacement) {\n firstFittingPlacement = fittingPlacement;\n return \"break\";\n }\n };\n\n for (var _i = numberOfChecks; _i > 0; _i--) {\n var _ret = _loop(_i);\n\n if (_ret === \"break\") break;\n }\n }\n\n if (state.placement !== firstFittingPlacement) {\n state.modifiersData[name]._skip = true;\n state.placement = firstFittingPlacement;\n state.reset = true;\n }\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'flip',\n enabled: true,\n phase: 'main',\n fn: flip,\n requiresIfExists: ['offset'],\n data: {\n _skip: false\n }\n};","import getVariation from \"./getVariation.js\";\nimport { variationPlacements, basePlacements, placements as allPlacements } from \"../enums.js\";\nimport detectOverflow from \"./detectOverflow.js\";\nimport getBasePlacement from \"./getBasePlacement.js\";\nexport default function computeAutoPlacement(state, options) {\n if (options === void 0) {\n options = {};\n }\n\n var _options = options,\n placement = _options.placement,\n boundary = _options.boundary,\n rootBoundary = _options.rootBoundary,\n padding = _options.padding,\n flipVariations = _options.flipVariations,\n _options$allowedAutoP = _options.allowedAutoPlacements,\n allowedAutoPlacements = _options$allowedAutoP === void 0 ? allPlacements : _options$allowedAutoP;\n var variation = getVariation(placement);\n var placements = variation ? flipVariations ? variationPlacements : variationPlacements.filter(function (placement) {\n return getVariation(placement) === variation;\n }) : basePlacements;\n var allowedPlacements = placements.filter(function (placement) {\n return allowedAutoPlacements.indexOf(placement) >= 0;\n });\n\n if (allowedPlacements.length === 0) {\n allowedPlacements = placements;\n } // $FlowFixMe[incompatible-type]: Flow seems to have problems with two array unions...\n\n\n var overflows = allowedPlacements.reduce(function (acc, placement) {\n acc[placement] = detectOverflow(state, {\n placement: placement,\n boundary: boundary,\n rootBoundary: rootBoundary,\n padding: padding\n })[getBasePlacement(placement)];\n return acc;\n }, {});\n return Object.keys(overflows).sort(function (a, b) {\n return overflows[a] - overflows[b];\n });\n}","import { top, bottom, left, right } from \"../enums.js\";\nimport detectOverflow from \"../utils/detectOverflow.js\";\n\nfunction getSideOffsets(overflow, rect, preventedOffsets) {\n if (preventedOffsets === void 0) {\n preventedOffsets = {\n x: 0,\n y: 0\n };\n }\n\n return {\n top: overflow.top - rect.height - preventedOffsets.y,\n right: overflow.right - rect.width + preventedOffsets.x,\n bottom: overflow.bottom - rect.height + preventedOffsets.y,\n left: overflow.left - rect.width - preventedOffsets.x\n };\n}\n\nfunction isAnySideFullyClipped(overflow) {\n return [top, right, bottom, left].some(function (side) {\n return overflow[side] >= 0;\n });\n}\n\nfunction hide(_ref) {\n var state = _ref.state,\n name = _ref.name;\n var referenceRect = state.rects.reference;\n var popperRect = state.rects.popper;\n var preventedOffsets = state.modifiersData.preventOverflow;\n var referenceOverflow = detectOverflow(state, {\n elementContext: 'reference'\n });\n var popperAltOverflow = detectOverflow(state, {\n altBoundary: true\n });\n var referenceClippingOffsets = getSideOffsets(referenceOverflow, referenceRect);\n var popperEscapeOffsets = getSideOffsets(popperAltOverflow, popperRect, preventedOffsets);\n var isReferenceHidden = isAnySideFullyClipped(referenceClippingOffsets);\n var hasPopperEscaped = isAnySideFullyClipped(popperEscapeOffsets);\n state.modifiersData[name] = {\n referenceClippingOffsets: referenceClippingOffsets,\n popperEscapeOffsets: popperEscapeOffsets,\n isReferenceHidden: isReferenceHidden,\n hasPopperEscaped: hasPopperEscaped\n };\n state.attributes.popper = Object.assign({}, state.attributes.popper, {\n 'data-popper-reference-hidden': isReferenceHidden,\n 'data-popper-escaped': hasPopperEscaped\n });\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'hide',\n enabled: true,\n phase: 'main',\n requiresIfExists: ['preventOverflow'],\n fn: hide\n};","import getBasePlacement from \"../utils/getBasePlacement.js\";\nimport { top, left, right, placements } from \"../enums.js\"; // eslint-disable-next-line import/no-unused-modules\n\nexport function distanceAndSkiddingToXY(placement, rects, offset) {\n var basePlacement = getBasePlacement(placement);\n var invertDistance = [left, top].indexOf(basePlacement) >= 0 ? -1 : 1;\n\n var _ref = typeof offset === 'function' ? offset(Object.assign({}, rects, {\n placement: placement\n })) : offset,\n skidding = _ref[0],\n distance = _ref[1];\n\n skidding = skidding || 0;\n distance = (distance || 0) * invertDistance;\n return [left, right].indexOf(basePlacement) >= 0 ? {\n x: distance,\n y: skidding\n } : {\n x: skidding,\n y: distance\n };\n}\n\nfunction offset(_ref2) {\n var state = _ref2.state,\n options = _ref2.options,\n name = _ref2.name;\n var _options$offset = options.offset,\n offset = _options$offset === void 0 ? [0, 0] : _options$offset;\n var data = placements.reduce(function (acc, placement) {\n acc[placement] = distanceAndSkiddingToXY(placement, state.rects, offset);\n return acc;\n }, {});\n var _data$state$placement = data[state.placement],\n x = _data$state$placement.x,\n y = _data$state$placement.y;\n\n if (state.modifiersData.popperOffsets != null) {\n state.modifiersData.popperOffsets.x += x;\n state.modifiersData.popperOffsets.y += y;\n }\n\n state.modifiersData[name] = data;\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'offset',\n enabled: true,\n phase: 'main',\n requires: ['popperOffsets'],\n fn: offset\n};","import computeOffsets from \"../utils/computeOffsets.js\";\n\nfunction popperOffsets(_ref) {\n var state = _ref.state,\n name = _ref.name;\n // Offsets are the actual position the popper needs to have to be\n // properly positioned near its reference element\n // This is the most basic placement, and will be adjusted by\n // the modifiers in the next step\n state.modifiersData[name] = computeOffsets({\n reference: state.rects.reference,\n element: state.rects.popper,\n strategy: 'absolute',\n placement: state.placement\n });\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'popperOffsets',\n enabled: true,\n phase: 'read',\n fn: popperOffsets,\n data: {}\n};","import { top, left, right, bottom, start } from \"../enums.js\";\nimport getBasePlacement from \"../utils/getBasePlacement.js\";\nimport getMainAxisFromPlacement from \"../utils/getMainAxisFromPlacement.js\";\nimport getAltAxis from \"../utils/getAltAxis.js\";\nimport { within, withinMaxClamp } from \"../utils/within.js\";\nimport getLayoutRect from \"../dom-utils/getLayoutRect.js\";\nimport getOffsetParent from \"../dom-utils/getOffsetParent.js\";\nimport detectOverflow from \"../utils/detectOverflow.js\";\nimport getVariation from \"../utils/getVariation.js\";\nimport getFreshSideObject from \"../utils/getFreshSideObject.js\";\nimport { min as mathMin, max as mathMax } from \"../utils/math.js\";\n\nfunction preventOverflow(_ref) {\n var state = _ref.state,\n options = _ref.options,\n name = _ref.name;\n var _options$mainAxis = options.mainAxis,\n checkMainAxis = _options$mainAxis === void 0 ? true : _options$mainAxis,\n _options$altAxis = options.altAxis,\n checkAltAxis = _options$altAxis === void 0 ? false : _options$altAxis,\n boundary = options.boundary,\n rootBoundary = options.rootBoundary,\n altBoundary = options.altBoundary,\n padding = options.padding,\n _options$tether = options.tether,\n tether = _options$tether === void 0 ? true : _options$tether,\n _options$tetherOffset = options.tetherOffset,\n tetherOffset = _options$tetherOffset === void 0 ? 0 : _options$tetherOffset;\n var overflow = detectOverflow(state, {\n boundary: boundary,\n rootBoundary: rootBoundary,\n padding: padding,\n altBoundary: altBoundary\n });\n var basePlacement = getBasePlacement(state.placement);\n var variation = getVariation(state.placement);\n var isBasePlacement = !variation;\n var mainAxis = getMainAxisFromPlacement(basePlacement);\n var altAxis = getAltAxis(mainAxis);\n var popperOffsets = state.modifiersData.popperOffsets;\n var referenceRect = state.rects.reference;\n var popperRect = state.rects.popper;\n var tetherOffsetValue = typeof tetherOffset === 'function' ? tetherOffset(Object.assign({}, state.rects, {\n placement: state.placement\n })) : tetherOffset;\n var normalizedTetherOffsetValue = typeof tetherOffsetValue === 'number' ? {\n mainAxis: tetherOffsetValue,\n altAxis: tetherOffsetValue\n } : Object.assign({\n mainAxis: 0,\n altAxis: 0\n }, tetherOffsetValue);\n var offsetModifierState = state.modifiersData.offset ? state.modifiersData.offset[state.placement] : null;\n var data = {\n x: 0,\n y: 0\n };\n\n if (!popperOffsets) {\n return;\n }\n\n if (checkMainAxis) {\n var _offsetModifierState$;\n\n var mainSide = mainAxis === 'y' ? top : left;\n var altSide = mainAxis === 'y' ? bottom : right;\n var len = mainAxis === 'y' ? 'height' : 'width';\n var offset = popperOffsets[mainAxis];\n var min = offset + overflow[mainSide];\n var max = offset - overflow[altSide];\n var additive = tether ? -popperRect[len] / 2 : 0;\n var minLen = variation === start ? referenceRect[len] : popperRect[len];\n var maxLen = variation === start ? -popperRect[len] : -referenceRect[len]; // We need to include the arrow in the calculation so the arrow doesn't go\n // outside the reference bounds\n\n var arrowElement = state.elements.arrow;\n var arrowRect = tether && arrowElement ? getLayoutRect(arrowElement) : {\n width: 0,\n height: 0\n };\n var arrowPaddingObject = state.modifiersData['arrow#persistent'] ? state.modifiersData['arrow#persistent'].padding : getFreshSideObject();\n var arrowPaddingMin = arrowPaddingObject[mainSide];\n var arrowPaddingMax = arrowPaddingObject[altSide]; // If the reference length is smaller than the arrow length, we don't want\n // to include its full size in the calculation. If the reference is small\n // and near the edge of a boundary, the popper can overflow even if the\n // reference is not overflowing as well (e.g. virtual elements with no\n // width or height)\n\n var arrowLen = within(0, referenceRect[len], arrowRect[len]);\n var minOffset = isBasePlacement ? referenceRect[len] / 2 - additive - arrowLen - arrowPaddingMin - normalizedTetherOffsetValue.mainAxis : minLen - arrowLen - arrowPaddingMin - normalizedTetherOffsetValue.mainAxis;\n var maxOffset = isBasePlacement ? -referenceRect[len] / 2 + additive + arrowLen + arrowPaddingMax + normalizedTetherOffsetValue.mainAxis : maxLen + arrowLen + arrowPaddingMax + normalizedTetherOffsetValue.mainAxis;\n var arrowOffsetParent = state.elements.arrow && getOffsetParent(state.elements.arrow);\n var clientOffset = arrowOffsetParent ? mainAxis === 'y' ? arrowOffsetParent.clientTop || 0 : arrowOffsetParent.clientLeft || 0 : 0;\n var offsetModifierValue = (_offsetModifierState$ = offsetModifierState == null ? void 0 : offsetModifierState[mainAxis]) != null ? _offsetModifierState$ : 0;\n var tetherMin = offset + minOffset - offsetModifierValue - clientOffset;\n var tetherMax = offset + maxOffset - offsetModifierValue;\n var preventedOffset = within(tether ? mathMin(min, tetherMin) : min, offset, tether ? mathMax(max, tetherMax) : max);\n popperOffsets[mainAxis] = preventedOffset;\n data[mainAxis] = preventedOffset - offset;\n }\n\n if (checkAltAxis) {\n var _offsetModifierState$2;\n\n var _mainSide = mainAxis === 'x' ? top : left;\n\n var _altSide = mainAxis === 'x' ? bottom : right;\n\n var _offset = popperOffsets[altAxis];\n\n var _len = altAxis === 'y' ? 'height' : 'width';\n\n var _min = _offset + overflow[_mainSide];\n\n var _max = _offset - overflow[_altSide];\n\n var isOriginSide = [top, left].indexOf(basePlacement) !== -1;\n\n var _offsetModifierValue = (_offsetModifierState$2 = offsetModifierState == null ? void 0 : offsetModifierState[altAxis]) != null ? _offsetModifierState$2 : 0;\n\n var _tetherMin = isOriginSide ? _min : _offset - referenceRect[_len] - popperRect[_len] - _offsetModifierValue + normalizedTetherOffsetValue.altAxis;\n\n var _tetherMax = isOriginSide ? _offset + referenceRect[_len] + popperRect[_len] - _offsetModifierValue - normalizedTetherOffsetValue.altAxis : _max;\n\n var _preventedOffset = tether && isOriginSide ? withinMaxClamp(_tetherMin, _offset, _tetherMax) : within(tether ? _tetherMin : _min, _offset, tether ? _tetherMax : _max);\n\n popperOffsets[altAxis] = _preventedOffset;\n data[altAxis] = _preventedOffset - _offset;\n }\n\n state.modifiersData[name] = data;\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'preventOverflow',\n enabled: true,\n phase: 'main',\n fn: preventOverflow,\n requiresIfExists: ['offset']\n};","export default function getAltAxis(axis) {\n return axis === 'x' ? 'y' : 'x';\n}","import getBoundingClientRect from \"./getBoundingClientRect.js\";\nimport getNodeScroll from \"./getNodeScroll.js\";\nimport getNodeName from \"./getNodeName.js\";\nimport { isHTMLElement } from \"./instanceOf.js\";\nimport getWindowScrollBarX from \"./getWindowScrollBarX.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport isScrollParent from \"./isScrollParent.js\";\nimport { round } from \"../utils/math.js\";\n\nfunction isElementScaled(element) {\n var rect = element.getBoundingClientRect();\n var scaleX = round(rect.width) / element.offsetWidth || 1;\n var scaleY = round(rect.height) / element.offsetHeight || 1;\n return scaleX !== 1 || scaleY !== 1;\n} // Returns the composite rect of an element relative to its offsetParent.\n// Composite means it takes into account transforms as well as layout.\n\n\nexport default function getCompositeRect(elementOrVirtualElement, offsetParent, isFixed) {\n if (isFixed === void 0) {\n isFixed = false;\n }\n\n var isOffsetParentAnElement = isHTMLElement(offsetParent);\n var offsetParentIsScaled = isHTMLElement(offsetParent) && isElementScaled(offsetParent);\n var documentElement = getDocumentElement(offsetParent);\n var rect = getBoundingClientRect(elementOrVirtualElement, offsetParentIsScaled, isFixed);\n var scroll = {\n scrollLeft: 0,\n scrollTop: 0\n };\n var offsets = {\n x: 0,\n y: 0\n };\n\n if (isOffsetParentAnElement || !isOffsetParentAnElement && !isFixed) {\n if (getNodeName(offsetParent) !== 'body' || // https://github.com/popperjs/popper-core/issues/1078\n isScrollParent(documentElement)) {\n scroll = getNodeScroll(offsetParent);\n }\n\n if (isHTMLElement(offsetParent)) {\n offsets = getBoundingClientRect(offsetParent, true);\n offsets.x += offsetParent.clientLeft;\n offsets.y += offsetParent.clientTop;\n } else if (documentElement) {\n offsets.x = getWindowScrollBarX(documentElement);\n }\n }\n\n return {\n x: rect.left + scroll.scrollLeft - offsets.x,\n y: rect.top + scroll.scrollTop - offsets.y,\n width: rect.width,\n height: rect.height\n };\n}","import getWindowScroll from \"./getWindowScroll.js\";\nimport getWindow from \"./getWindow.js\";\nimport { isHTMLElement } from \"./instanceOf.js\";\nimport getHTMLElementScroll from \"./getHTMLElementScroll.js\";\nexport default function getNodeScroll(node) {\n if (node === getWindow(node) || !isHTMLElement(node)) {\n return getWindowScroll(node);\n } else {\n return getHTMLElementScroll(node);\n }\n}","export default function getHTMLElementScroll(element) {\n return {\n scrollLeft: element.scrollLeft,\n scrollTop: element.scrollTop\n };\n}","import { modifierPhases } from \"../enums.js\"; // source: https://stackoverflow.com/questions/49875255\n\nfunction order(modifiers) {\n var map = new Map();\n var visited = new Set();\n var result = [];\n modifiers.forEach(function (modifier) {\n map.set(modifier.name, modifier);\n }); // On visiting object, check for its dependencies and visit them recursively\n\n function sort(modifier) {\n visited.add(modifier.name);\n var requires = [].concat(modifier.requires || [], modifier.requiresIfExists || []);\n requires.forEach(function (dep) {\n if (!visited.has(dep)) {\n var depModifier = map.get(dep);\n\n if (depModifier) {\n sort(depModifier);\n }\n }\n });\n result.push(modifier);\n }\n\n modifiers.forEach(function (modifier) {\n if (!visited.has(modifier.name)) {\n // check for visited object\n sort(modifier);\n }\n });\n return result;\n}\n\nexport default function orderModifiers(modifiers) {\n // order based on dependencies\n var orderedModifiers = order(modifiers); // order based on phase\n\n return modifierPhases.reduce(function (acc, phase) {\n return acc.concat(orderedModifiers.filter(function (modifier) {\n return modifier.phase === phase;\n }));\n }, []);\n}","import getCompositeRect from \"./dom-utils/getCompositeRect.js\";\nimport getLayoutRect from \"./dom-utils/getLayoutRect.js\";\nimport listScrollParents from \"./dom-utils/listScrollParents.js\";\nimport getOffsetParent from \"./dom-utils/getOffsetParent.js\";\nimport orderModifiers from \"./utils/orderModifiers.js\";\nimport debounce from \"./utils/debounce.js\";\nimport mergeByName from \"./utils/mergeByName.js\";\nimport detectOverflow from \"./utils/detectOverflow.js\";\nimport { isElement } from \"./dom-utils/instanceOf.js\";\nvar DEFAULT_OPTIONS = {\n placement: 'bottom',\n modifiers: [],\n strategy: 'absolute'\n};\n\nfunction areValidElements() {\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n\n return !args.some(function (element) {\n return !(element && typeof element.getBoundingClientRect === 'function');\n });\n}\n\nexport function popperGenerator(generatorOptions) {\n if (generatorOptions === void 0) {\n generatorOptions = {};\n }\n\n var _generatorOptions = generatorOptions,\n _generatorOptions$def = _generatorOptions.defaultModifiers,\n defaultModifiers = _generatorOptions$def === void 0 ? [] : _generatorOptions$def,\n _generatorOptions$def2 = _generatorOptions.defaultOptions,\n defaultOptions = _generatorOptions$def2 === void 0 ? DEFAULT_OPTIONS : _generatorOptions$def2;\n return function createPopper(reference, popper, options) {\n if (options === void 0) {\n options = defaultOptions;\n }\n\n var state = {\n placement: 'bottom',\n orderedModifiers: [],\n options: Object.assign({}, DEFAULT_OPTIONS, defaultOptions),\n modifiersData: {},\n elements: {\n reference: reference,\n popper: popper\n },\n attributes: {},\n styles: {}\n };\n var effectCleanupFns = [];\n var isDestroyed = false;\n var instance = {\n state: state,\n setOptions: function setOptions(setOptionsAction) {\n var options = typeof setOptionsAction === 'function' ? setOptionsAction(state.options) : setOptionsAction;\n cleanupModifierEffects();\n state.options = Object.assign({}, defaultOptions, state.options, options);\n state.scrollParents = {\n reference: isElement(reference) ? listScrollParents(reference) : reference.contextElement ? listScrollParents(reference.contextElement) : [],\n popper: listScrollParents(popper)\n }; // Orders the modifiers based on their dependencies and `phase`\n // properties\n\n var orderedModifiers = orderModifiers(mergeByName([].concat(defaultModifiers, state.options.modifiers))); // Strip out disabled modifiers\n\n state.orderedModifiers = orderedModifiers.filter(function (m) {\n return m.enabled;\n });\n runModifierEffects();\n return instance.update();\n },\n // Sync update – it will always be executed, even if not necessary. This\n // is useful for low frequency updates where sync behavior simplifies the\n // logic.\n // For high frequency updates (e.g. `resize` and `scroll` events), always\n // prefer the async Popper#update method\n forceUpdate: function forceUpdate() {\n if (isDestroyed) {\n return;\n }\n\n var _state$elements = state.elements,\n reference = _state$elements.reference,\n popper = _state$elements.popper; // Don't proceed if `reference` or `popper` are not valid elements\n // anymore\n\n if (!areValidElements(reference, popper)) {\n return;\n } // Store the reference and popper rects to be read by modifiers\n\n\n state.rects = {\n reference: getCompositeRect(reference, getOffsetParent(popper), state.options.strategy === 'fixed'),\n popper: getLayoutRect(popper)\n }; // Modifiers have the ability to reset the current update cycle. The\n // most common use case for this is the `flip` modifier changing the\n // placement, which then needs to re-run all the modifiers, because the\n // logic was previously ran for the previous placement and is therefore\n // stale/incorrect\n\n state.reset = false;\n state.placement = state.options.placement; // On each update cycle, the `modifiersData` property for each modifier\n // is filled with the initial data specified by the modifier. This means\n // it doesn't persist and is fresh on each update.\n // To ensure persistent data, use `${name}#persistent`\n\n state.orderedModifiers.forEach(function (modifier) {\n return state.modifiersData[modifier.name] = Object.assign({}, modifier.data);\n });\n\n for (var index = 0; index < state.orderedModifiers.length; index++) {\n if (state.reset === true) {\n state.reset = false;\n index = -1;\n continue;\n }\n\n var _state$orderedModifie = state.orderedModifiers[index],\n fn = _state$orderedModifie.fn,\n _state$orderedModifie2 = _state$orderedModifie.options,\n _options = _state$orderedModifie2 === void 0 ? {} : _state$orderedModifie2,\n name = _state$orderedModifie.name;\n\n if (typeof fn === 'function') {\n state = fn({\n state: state,\n options: _options,\n name: name,\n instance: instance\n }) || state;\n }\n }\n },\n // Async and optimistically optimized update – it will not be executed if\n // not necessary (debounced to run at most once-per-tick)\n update: debounce(function () {\n return new Promise(function (resolve) {\n instance.forceUpdate();\n resolve(state);\n });\n }),\n destroy: function destroy() {\n cleanupModifierEffects();\n isDestroyed = true;\n }\n };\n\n if (!areValidElements(reference, popper)) {\n return instance;\n }\n\n instance.setOptions(options).then(function (state) {\n if (!isDestroyed && options.onFirstUpdate) {\n options.onFirstUpdate(state);\n }\n }); // Modifiers have the ability to execute arbitrary code before the first\n // update cycle runs. They will be executed in the same order as the update\n // cycle. This is useful when a modifier adds some persistent data that\n // other modifiers need to use, but the modifier is run after the dependent\n // one.\n\n function runModifierEffects() {\n state.orderedModifiers.forEach(function (_ref) {\n var name = _ref.name,\n _ref$options = _ref.options,\n options = _ref$options === void 0 ? {} : _ref$options,\n effect = _ref.effect;\n\n if (typeof effect === 'function') {\n var cleanupFn = effect({\n state: state,\n name: name,\n instance: instance,\n options: options\n });\n\n var noopFn = function noopFn() {};\n\n effectCleanupFns.push(cleanupFn || noopFn);\n }\n });\n }\n\n function cleanupModifierEffects() {\n effectCleanupFns.forEach(function (fn) {\n return fn();\n });\n effectCleanupFns = [];\n }\n\n return instance;\n };\n}\nexport var createPopper = /*#__PURE__*/popperGenerator(); // eslint-disable-next-line import/no-unused-modules\n\nexport { detectOverflow };","export default function debounce(fn) {\n var pending;\n return function () {\n if (!pending) {\n pending = new Promise(function (resolve) {\n Promise.resolve().then(function () {\n pending = undefined;\n resolve(fn());\n });\n });\n }\n\n return pending;\n };\n}","export default function mergeByName(modifiers) {\n var merged = modifiers.reduce(function (merged, current) {\n var existing = merged[current.name];\n merged[current.name] = existing ? Object.assign({}, existing, current, {\n options: Object.assign({}, existing.options, current.options),\n data: Object.assign({}, existing.data, current.data)\n }) : current;\n return merged;\n }, {}); // IE11 does not support Object.values\n\n return Object.keys(merged).map(function (key) {\n return merged[key];\n });\n}","import { popperGenerator, detectOverflow } from \"./createPopper.js\";\nimport eventListeners from \"./modifiers/eventListeners.js\";\nimport popperOffsets from \"./modifiers/popperOffsets.js\";\nimport computeStyles from \"./modifiers/computeStyles.js\";\nimport applyStyles from \"./modifiers/applyStyles.js\";\nimport offset from \"./modifiers/offset.js\";\nimport flip from \"./modifiers/flip.js\";\nimport preventOverflow from \"./modifiers/preventOverflow.js\";\nimport arrow from \"./modifiers/arrow.js\";\nimport hide from \"./modifiers/hide.js\";\nvar defaultModifiers = [eventListeners, popperOffsets, computeStyles, applyStyles, offset, flip, preventOverflow, arrow, hide];\nvar createPopper = /*#__PURE__*/popperGenerator({\n defaultModifiers: defaultModifiers\n}); // eslint-disable-next-line import/no-unused-modules\n\nexport { createPopper, popperGenerator, defaultModifiers, detectOverflow }; // eslint-disable-next-line import/no-unused-modules\n\nexport { createPopper as createPopperLite } from \"./popper-lite.js\"; // eslint-disable-next-line import/no-unused-modules\n\nexport * from \"./modifiers/index.js\";","import { popperGenerator, detectOverflow } from \"./createPopper.js\";\nimport eventListeners from \"./modifiers/eventListeners.js\";\nimport popperOffsets from \"./modifiers/popperOffsets.js\";\nimport computeStyles from \"./modifiers/computeStyles.js\";\nimport applyStyles from \"./modifiers/applyStyles.js\";\nvar defaultModifiers = [eventListeners, popperOffsets, computeStyles, applyStyles];\nvar createPopper = /*#__PURE__*/popperGenerator({\n defaultModifiers: defaultModifiers\n}); // eslint-disable-next-line import/no-unused-modules\n\nexport { createPopper, popperGenerator, defaultModifiers, detectOverflow };","/*!\n * Bootstrap v5.3.3 (https://getbootstrap.com/)\n * Copyright 2011-2024 The Bootstrap Authors (https://github.com/twbs/bootstrap/graphs/contributors)\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n */\nimport * as Popper from '@popperjs/core';\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dom/data.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n/**\n * Constants\n */\n\nconst elementMap = new Map();\nconst Data = {\n set(element, key, instance) {\n if (!elementMap.has(element)) {\n elementMap.set(element, new Map());\n }\n const instanceMap = elementMap.get(element);\n\n // make it clear we only want one instance per element\n // can be removed later when multiple key/instances are fine to be used\n if (!instanceMap.has(key) && instanceMap.size !== 0) {\n // eslint-disable-next-line no-console\n console.error(`Bootstrap doesn't allow more than one instance per element. Bound instance: ${Array.from(instanceMap.keys())[0]}.`);\n return;\n }\n instanceMap.set(key, instance);\n },\n get(element, key) {\n if (elementMap.has(element)) {\n return elementMap.get(element).get(key) || null;\n }\n return null;\n },\n remove(element, key) {\n if (!elementMap.has(element)) {\n return;\n }\n const instanceMap = elementMap.get(element);\n instanceMap.delete(key);\n\n // free up element references if there are no instances left for an element\n if (instanceMap.size === 0) {\n elementMap.delete(element);\n }\n }\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/index.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\nconst MAX_UID = 1000000;\nconst MILLISECONDS_MULTIPLIER = 1000;\nconst TRANSITION_END = 'transitionend';\n\n/**\n * Properly escape IDs selectors to handle weird IDs\n * @param {string} selector\n * @returns {string}\n */\nconst parseSelector = selector => {\n if (selector && window.CSS && window.CSS.escape) {\n // document.querySelector needs escaping to handle IDs (html5+) containing for instance /\n selector = selector.replace(/#([^\\s\"#']+)/g, (match, id) => `#${CSS.escape(id)}`);\n }\n return selector;\n};\n\n// Shout-out Angus Croll (https://goo.gl/pxwQGp)\nconst toType = object => {\n if (object === null || object === undefined) {\n return `${object}`;\n }\n return Object.prototype.toString.call(object).match(/\\s([a-z]+)/i)[1].toLowerCase();\n};\n\n/**\n * Public Util API\n */\n\nconst getUID = prefix => {\n do {\n prefix += Math.floor(Math.random() * MAX_UID);\n } while (document.getElementById(prefix));\n return prefix;\n};\nconst getTransitionDurationFromElement = element => {\n if (!element) {\n return 0;\n }\n\n // Get transition-duration of the element\n let {\n transitionDuration,\n transitionDelay\n } = window.getComputedStyle(element);\n const floatTransitionDuration = Number.parseFloat(transitionDuration);\n const floatTransitionDelay = Number.parseFloat(transitionDelay);\n\n // Return 0 if element or transition duration is not found\n if (!floatTransitionDuration && !floatTransitionDelay) {\n return 0;\n }\n\n // If multiple durations are defined, take the first\n transitionDuration = transitionDuration.split(',')[0];\n transitionDelay = transitionDelay.split(',')[0];\n return (Number.parseFloat(transitionDuration) + Number.parseFloat(transitionDelay)) * MILLISECONDS_MULTIPLIER;\n};\nconst triggerTransitionEnd = element => {\n element.dispatchEvent(new Event(TRANSITION_END));\n};\nconst isElement = object => {\n if (!object || typeof object !== 'object') {\n return false;\n }\n if (typeof object.jquery !== 'undefined') {\n object = object[0];\n }\n return typeof object.nodeType !== 'undefined';\n};\nconst getElement = object => {\n // it's a jQuery object or a node element\n if (isElement(object)) {\n return object.jquery ? object[0] : object;\n }\n if (typeof object === 'string' && object.length > 0) {\n return document.querySelector(parseSelector(object));\n }\n return null;\n};\nconst isVisible = element => {\n if (!isElement(element) || element.getClientRects().length === 0) {\n return false;\n }\n const elementIsVisible = getComputedStyle(element).getPropertyValue('visibility') === 'visible';\n // Handle `details` element as its content may falsie appear visible when it is closed\n const closedDetails = element.closest('details:not([open])');\n if (!closedDetails) {\n return elementIsVisible;\n }\n if (closedDetails !== element) {\n const summary = element.closest('summary');\n if (summary && summary.parentNode !== closedDetails) {\n return false;\n }\n if (summary === null) {\n return false;\n }\n }\n return elementIsVisible;\n};\nconst isDisabled = element => {\n if (!element || element.nodeType !== Node.ELEMENT_NODE) {\n return true;\n }\n if (element.classList.contains('disabled')) {\n return true;\n }\n if (typeof element.disabled !== 'undefined') {\n return element.disabled;\n }\n return element.hasAttribute('disabled') && element.getAttribute('disabled') !== 'false';\n};\nconst findShadowRoot = element => {\n if (!document.documentElement.attachShadow) {\n return null;\n }\n\n // Can find the shadow root otherwise it'll return the document\n if (typeof element.getRootNode === 'function') {\n const root = element.getRootNode();\n return root instanceof ShadowRoot ? root : null;\n }\n if (element instanceof ShadowRoot) {\n return element;\n }\n\n // when we don't find a shadow root\n if (!element.parentNode) {\n return null;\n }\n return findShadowRoot(element.parentNode);\n};\nconst noop = () => {};\n\n/**\n * Trick to restart an element's animation\n *\n * @param {HTMLElement} element\n * @return void\n *\n * @see https://www.charistheo.io/blog/2021/02/restart-a-css-animation-with-javascript/#restarting-a-css-animation\n */\nconst reflow = element => {\n element.offsetHeight; // eslint-disable-line no-unused-expressions\n};\nconst getjQuery = () => {\n if (window.jQuery && !document.body.hasAttribute('data-bs-no-jquery')) {\n return window.jQuery;\n }\n return null;\n};\nconst DOMContentLoadedCallbacks = [];\nconst onDOMContentLoaded = callback => {\n if (document.readyState === 'loading') {\n // add listener on the first call when the document is in loading state\n if (!DOMContentLoadedCallbacks.length) {\n document.addEventListener('DOMContentLoaded', () => {\n for (const callback of DOMContentLoadedCallbacks) {\n callback();\n }\n });\n }\n DOMContentLoadedCallbacks.push(callback);\n } else {\n callback();\n }\n};\nconst isRTL = () => document.documentElement.dir === 'rtl';\nconst defineJQueryPlugin = plugin => {\n onDOMContentLoaded(() => {\n const $ = getjQuery();\n /* istanbul ignore if */\n if ($) {\n const name = plugin.NAME;\n const JQUERY_NO_CONFLICT = $.fn[name];\n $.fn[name] = plugin.jQueryInterface;\n $.fn[name].Constructor = plugin;\n $.fn[name].noConflict = () => {\n $.fn[name] = JQUERY_NO_CONFLICT;\n return plugin.jQueryInterface;\n };\n }\n });\n};\nconst execute = (possibleCallback, args = [], defaultValue = possibleCallback) => {\n return typeof possibleCallback === 'function' ? possibleCallback(...args) : defaultValue;\n};\nconst executeAfterTransition = (callback, transitionElement, waitForTransition = true) => {\n if (!waitForTransition) {\n execute(callback);\n return;\n }\n const durationPadding = 5;\n const emulatedDuration = getTransitionDurationFromElement(transitionElement) + durationPadding;\n let called = false;\n const handler = ({\n target\n }) => {\n if (target !== transitionElement) {\n return;\n }\n called = true;\n transitionElement.removeEventListener(TRANSITION_END, handler);\n execute(callback);\n };\n transitionElement.addEventListener(TRANSITION_END, handler);\n setTimeout(() => {\n if (!called) {\n triggerTransitionEnd(transitionElement);\n }\n }, emulatedDuration);\n};\n\n/**\n * Return the previous/next element of a list.\n *\n * @param {array} list The list of elements\n * @param activeElement The active element\n * @param shouldGetNext Choose to get next or previous element\n * @param isCycleAllowed\n * @return {Element|elem} The proper element\n */\nconst getNextActiveElement = (list, activeElement, shouldGetNext, isCycleAllowed) => {\n const listLength = list.length;\n let index = list.indexOf(activeElement);\n\n // if the element does not exist in the list return an element\n // depending on the direction and if cycle is allowed\n if (index === -1) {\n return !shouldGetNext && isCycleAllowed ? list[listLength - 1] : list[0];\n }\n index += shouldGetNext ? 1 : -1;\n if (isCycleAllowed) {\n index = (index + listLength) % listLength;\n }\n return list[Math.max(0, Math.min(index, listLength - 1))];\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dom/event-handler.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst namespaceRegex = /[^.]*(?=\\..*)\\.|.*/;\nconst stripNameRegex = /\\..*/;\nconst stripUidRegex = /::\\d+$/;\nconst eventRegistry = {}; // Events storage\nlet uidEvent = 1;\nconst customEvents = {\n mouseenter: 'mouseover',\n mouseleave: 'mouseout'\n};\nconst nativeEvents = new Set(['click', 'dblclick', 'mouseup', 'mousedown', 'contextmenu', 'mousewheel', 'DOMMouseScroll', 'mouseover', 'mouseout', 'mousemove', 'selectstart', 'selectend', 'keydown', 'keypress', 'keyup', 'orientationchange', 'touchstart', 'touchmove', 'touchend', 'touchcancel', 'pointerdown', 'pointermove', 'pointerup', 'pointerleave', 'pointercancel', 'gesturestart', 'gesturechange', 'gestureend', 'focus', 'blur', 'change', 'reset', 'select', 'submit', 'focusin', 'focusout', 'load', 'unload', 'beforeunload', 'resize', 'move', 'DOMContentLoaded', 'readystatechange', 'error', 'abort', 'scroll']);\n\n/**\n * Private methods\n */\n\nfunction makeEventUid(element, uid) {\n return uid && `${uid}::${uidEvent++}` || element.uidEvent || uidEvent++;\n}\nfunction getElementEvents(element) {\n const uid = makeEventUid(element);\n element.uidEvent = uid;\n eventRegistry[uid] = eventRegistry[uid] || {};\n return eventRegistry[uid];\n}\nfunction bootstrapHandler(element, fn) {\n return function handler(event) {\n hydrateObj(event, {\n delegateTarget: element\n });\n if (handler.oneOff) {\n EventHandler.off(element, event.type, fn);\n }\n return fn.apply(element, [event]);\n };\n}\nfunction bootstrapDelegationHandler(element, selector, fn) {\n return function handler(event) {\n const domElements = element.querySelectorAll(selector);\n for (let {\n target\n } = event; target && target !== this; target = target.parentNode) {\n for (const domElement of domElements) {\n if (domElement !== target) {\n continue;\n }\n hydrateObj(event, {\n delegateTarget: target\n });\n if (handler.oneOff) {\n EventHandler.off(element, event.type, selector, fn);\n }\n return fn.apply(target, [event]);\n }\n }\n };\n}\nfunction findHandler(events, callable, delegationSelector = null) {\n return Object.values(events).find(event => event.callable === callable && event.delegationSelector === delegationSelector);\n}\nfunction normalizeParameters(originalTypeEvent, handler, delegationFunction) {\n const isDelegated = typeof handler === 'string';\n // TODO: tooltip passes `false` instead of selector, so we need to check\n const callable = isDelegated ? delegationFunction : handler || delegationFunction;\n let typeEvent = getTypeEvent(originalTypeEvent);\n if (!nativeEvents.has(typeEvent)) {\n typeEvent = originalTypeEvent;\n }\n return [isDelegated, callable, typeEvent];\n}\nfunction addHandler(element, originalTypeEvent, handler, delegationFunction, oneOff) {\n if (typeof originalTypeEvent !== 'string' || !element) {\n return;\n }\n let [isDelegated, callable, typeEvent] = normalizeParameters(originalTypeEvent, handler, delegationFunction);\n\n // in case of mouseenter or mouseleave wrap the handler within a function that checks for its DOM position\n // this prevents the handler from being dispatched the same way as mouseover or mouseout does\n if (originalTypeEvent in customEvents) {\n const wrapFunction = fn => {\n return function (event) {\n if (!event.relatedTarget || event.relatedTarget !== event.delegateTarget && !event.delegateTarget.contains(event.relatedTarget)) {\n return fn.call(this, event);\n }\n };\n };\n callable = wrapFunction(callable);\n }\n const events = getElementEvents(element);\n const handlers = events[typeEvent] || (events[typeEvent] = {});\n const previousFunction = findHandler(handlers, callable, isDelegated ? handler : null);\n if (previousFunction) {\n previousFunction.oneOff = previousFunction.oneOff && oneOff;\n return;\n }\n const uid = makeEventUid(callable, originalTypeEvent.replace(namespaceRegex, ''));\n const fn = isDelegated ? bootstrapDelegationHandler(element, handler, callable) : bootstrapHandler(element, callable);\n fn.delegationSelector = isDelegated ? handler : null;\n fn.callable = callable;\n fn.oneOff = oneOff;\n fn.uidEvent = uid;\n handlers[uid] = fn;\n element.addEventListener(typeEvent, fn, isDelegated);\n}\nfunction removeHandler(element, events, typeEvent, handler, delegationSelector) {\n const fn = findHandler(events[typeEvent], handler, delegationSelector);\n if (!fn) {\n return;\n }\n element.removeEventListener(typeEvent, fn, Boolean(delegationSelector));\n delete events[typeEvent][fn.uidEvent];\n}\nfunction removeNamespacedHandlers(element, events, typeEvent, namespace) {\n const storeElementEvent = events[typeEvent] || {};\n for (const [handlerKey, event] of Object.entries(storeElementEvent)) {\n if (handlerKey.includes(namespace)) {\n removeHandler(element, events, typeEvent, event.callable, event.delegationSelector);\n }\n }\n}\nfunction getTypeEvent(event) {\n // allow to get the native events from namespaced events ('click.bs.button' --> 'click')\n event = event.replace(stripNameRegex, '');\n return customEvents[event] || event;\n}\nconst EventHandler = {\n on(element, event, handler, delegationFunction) {\n addHandler(element, event, handler, delegationFunction, false);\n },\n one(element, event, handler, delegationFunction) {\n addHandler(element, event, handler, delegationFunction, true);\n },\n off(element, originalTypeEvent, handler, delegationFunction) {\n if (typeof originalTypeEvent !== 'string' || !element) {\n return;\n }\n const [isDelegated, callable, typeEvent] = normalizeParameters(originalTypeEvent, handler, delegationFunction);\n const inNamespace = typeEvent !== originalTypeEvent;\n const events = getElementEvents(element);\n const storeElementEvent = events[typeEvent] || {};\n const isNamespace = originalTypeEvent.startsWith('.');\n if (typeof callable !== 'undefined') {\n // Simplest case: handler is passed, remove that listener ONLY.\n if (!Object.keys(storeElementEvent).length) {\n return;\n }\n removeHandler(element, events, typeEvent, callable, isDelegated ? handler : null);\n return;\n }\n if (isNamespace) {\n for (const elementEvent of Object.keys(events)) {\n removeNamespacedHandlers(element, events, elementEvent, originalTypeEvent.slice(1));\n }\n }\n for (const [keyHandlers, event] of Object.entries(storeElementEvent)) {\n const handlerKey = keyHandlers.replace(stripUidRegex, '');\n if (!inNamespace || originalTypeEvent.includes(handlerKey)) {\n removeHandler(element, events, typeEvent, event.callable, event.delegationSelector);\n }\n }\n },\n trigger(element, event, args) {\n if (typeof event !== 'string' || !element) {\n return null;\n }\n const $ = getjQuery();\n const typeEvent = getTypeEvent(event);\n const inNamespace = event !== typeEvent;\n let jQueryEvent = null;\n let bubbles = true;\n let nativeDispatch = true;\n let defaultPrevented = false;\n if (inNamespace && $) {\n jQueryEvent = $.Event(event, args);\n $(element).trigger(jQueryEvent);\n bubbles = !jQueryEvent.isPropagationStopped();\n nativeDispatch = !jQueryEvent.isImmediatePropagationStopped();\n defaultPrevented = jQueryEvent.isDefaultPrevented();\n }\n const evt = hydrateObj(new Event(event, {\n bubbles,\n cancelable: true\n }), args);\n if (defaultPrevented) {\n evt.preventDefault();\n }\n if (nativeDispatch) {\n element.dispatchEvent(evt);\n }\n if (evt.defaultPrevented && jQueryEvent) {\n jQueryEvent.preventDefault();\n }\n return evt;\n }\n};\nfunction hydrateObj(obj, meta = {}) {\n for (const [key, value] of Object.entries(meta)) {\n try {\n obj[key] = value;\n } catch (_unused) {\n Object.defineProperty(obj, key, {\n configurable: true,\n get() {\n return value;\n }\n });\n }\n }\n return obj;\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dom/manipulator.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\nfunction normalizeData(value) {\n if (value === 'true') {\n return true;\n }\n if (value === 'false') {\n return false;\n }\n if (value === Number(value).toString()) {\n return Number(value);\n }\n if (value === '' || value === 'null') {\n return null;\n }\n if (typeof value !== 'string') {\n return value;\n }\n try {\n return JSON.parse(decodeURIComponent(value));\n } catch (_unused) {\n return value;\n }\n}\nfunction normalizeDataKey(key) {\n return key.replace(/[A-Z]/g, chr => `-${chr.toLowerCase()}`);\n}\nconst Manipulator = {\n setDataAttribute(element, key, value) {\n element.setAttribute(`data-bs-${normalizeDataKey(key)}`, value);\n },\n removeDataAttribute(element, key) {\n element.removeAttribute(`data-bs-${normalizeDataKey(key)}`);\n },\n getDataAttributes(element) {\n if (!element) {\n return {};\n }\n const attributes = {};\n const bsKeys = Object.keys(element.dataset).filter(key => key.startsWith('bs') && !key.startsWith('bsConfig'));\n for (const key of bsKeys) {\n let pureKey = key.replace(/^bs/, '');\n pureKey = pureKey.charAt(0).toLowerCase() + pureKey.slice(1, pureKey.length);\n attributes[pureKey] = normalizeData(element.dataset[key]);\n }\n return attributes;\n },\n getDataAttribute(element, key) {\n return normalizeData(element.getAttribute(`data-bs-${normalizeDataKey(key)}`));\n }\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/config.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Class definition\n */\n\nclass Config {\n // Getters\n static get Default() {\n return {};\n }\n static get DefaultType() {\n return {};\n }\n static get NAME() {\n throw new Error('You have to implement the static method \"NAME\", for each component!');\n }\n _getConfig(config) {\n config = this._mergeConfigObj(config);\n config = this._configAfterMerge(config);\n this._typeCheckConfig(config);\n return config;\n }\n _configAfterMerge(config) {\n return config;\n }\n _mergeConfigObj(config, element) {\n const jsonConfig = isElement(element) ? Manipulator.getDataAttribute(element, 'config') : {}; // try to parse\n\n return {\n ...this.constructor.Default,\n ...(typeof jsonConfig === 'object' ? jsonConfig : {}),\n ...(isElement(element) ? Manipulator.getDataAttributes(element) : {}),\n ...(typeof config === 'object' ? config : {})\n };\n }\n _typeCheckConfig(config, configTypes = this.constructor.DefaultType) {\n for (const [property, expectedTypes] of Object.entries(configTypes)) {\n const value = config[property];\n const valueType = isElement(value) ? 'element' : toType(value);\n if (!new RegExp(expectedTypes).test(valueType)) {\n throw new TypeError(`${this.constructor.NAME.toUpperCase()}: Option \"${property}\" provided type \"${valueType}\" but expected type \"${expectedTypes}\".`);\n }\n }\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap base-component.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst VERSION = '5.3.3';\n\n/**\n * Class definition\n */\n\nclass BaseComponent extends Config {\n constructor(element, config) {\n super();\n element = getElement(element);\n if (!element) {\n return;\n }\n this._element = element;\n this._config = this._getConfig(config);\n Data.set(this._element, this.constructor.DATA_KEY, this);\n }\n\n // Public\n dispose() {\n Data.remove(this._element, this.constructor.DATA_KEY);\n EventHandler.off(this._element, this.constructor.EVENT_KEY);\n for (const propertyName of Object.getOwnPropertyNames(this)) {\n this[propertyName] = null;\n }\n }\n _queueCallback(callback, element, isAnimated = true) {\n executeAfterTransition(callback, element, isAnimated);\n }\n _getConfig(config) {\n config = this._mergeConfigObj(config, this._element);\n config = this._configAfterMerge(config);\n this._typeCheckConfig(config);\n return config;\n }\n\n // Static\n static getInstance(element) {\n return Data.get(getElement(element), this.DATA_KEY);\n }\n static getOrCreateInstance(element, config = {}) {\n return this.getInstance(element) || new this(element, typeof config === 'object' ? config : null);\n }\n static get VERSION() {\n return VERSION;\n }\n static get DATA_KEY() {\n return `bs.${this.NAME}`;\n }\n static get EVENT_KEY() {\n return `.${this.DATA_KEY}`;\n }\n static eventName(name) {\n return `${name}${this.EVENT_KEY}`;\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dom/selector-engine.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\nconst getSelector = element => {\n let selector = element.getAttribute('data-bs-target');\n if (!selector || selector === '#') {\n let hrefAttribute = element.getAttribute('href');\n\n // The only valid content that could double as a selector are IDs or classes,\n // so everything starting with `#` or `.`. If a \"real\" URL is used as the selector,\n // `document.querySelector` will rightfully complain it is invalid.\n // See https://github.com/twbs/bootstrap/issues/32273\n if (!hrefAttribute || !hrefAttribute.includes('#') && !hrefAttribute.startsWith('.')) {\n return null;\n }\n\n // Just in case some CMS puts out a full URL with the anchor appended\n if (hrefAttribute.includes('#') && !hrefAttribute.startsWith('#')) {\n hrefAttribute = `#${hrefAttribute.split('#')[1]}`;\n }\n selector = hrefAttribute && hrefAttribute !== '#' ? hrefAttribute.trim() : null;\n }\n return selector ? selector.split(',').map(sel => parseSelector(sel)).join(',') : null;\n};\nconst SelectorEngine = {\n find(selector, element = document.documentElement) {\n return [].concat(...Element.prototype.querySelectorAll.call(element, selector));\n },\n findOne(selector, element = document.documentElement) {\n return Element.prototype.querySelector.call(element, selector);\n },\n children(element, selector) {\n return [].concat(...element.children).filter(child => child.matches(selector));\n },\n parents(element, selector) {\n const parents = [];\n let ancestor = element.parentNode.closest(selector);\n while (ancestor) {\n parents.push(ancestor);\n ancestor = ancestor.parentNode.closest(selector);\n }\n return parents;\n },\n prev(element, selector) {\n let previous = element.previousElementSibling;\n while (previous) {\n if (previous.matches(selector)) {\n return [previous];\n }\n previous = previous.previousElementSibling;\n }\n return [];\n },\n // TODO: this is now unused; remove later along with prev()\n next(element, selector) {\n let next = element.nextElementSibling;\n while (next) {\n if (next.matches(selector)) {\n return [next];\n }\n next = next.nextElementSibling;\n }\n return [];\n },\n focusableChildren(element) {\n const focusables = ['a', 'button', 'input', 'textarea', 'select', 'details', '[tabindex]', '[contenteditable=\"true\"]'].map(selector => `${selector}:not([tabindex^=\"-\"])`).join(',');\n return this.find(focusables, element).filter(el => !isDisabled(el) && isVisible(el));\n },\n getSelectorFromElement(element) {\n const selector = getSelector(element);\n if (selector) {\n return SelectorEngine.findOne(selector) ? selector : null;\n }\n return null;\n },\n getElementFromSelector(element) {\n const selector = getSelector(element);\n return selector ? SelectorEngine.findOne(selector) : null;\n },\n getMultipleElementsFromSelector(element) {\n const selector = getSelector(element);\n return selector ? SelectorEngine.find(selector) : [];\n }\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/component-functions.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\nconst enableDismissTrigger = (component, method = 'hide') => {\n const clickEvent = `click.dismiss${component.EVENT_KEY}`;\n const name = component.NAME;\n EventHandler.on(document, clickEvent, `[data-bs-dismiss=\"${name}\"]`, function (event) {\n if (['A', 'AREA'].includes(this.tagName)) {\n event.preventDefault();\n }\n if (isDisabled(this)) {\n return;\n }\n const target = SelectorEngine.getElementFromSelector(this) || this.closest(`.${name}`);\n const instance = component.getOrCreateInstance(target);\n\n // Method argument is left, for Alert and only, as it doesn't implement the 'hide' method\n instance[method]();\n });\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap alert.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$f = 'alert';\nconst DATA_KEY$a = 'bs.alert';\nconst EVENT_KEY$b = `.${DATA_KEY$a}`;\nconst EVENT_CLOSE = `close${EVENT_KEY$b}`;\nconst EVENT_CLOSED = `closed${EVENT_KEY$b}`;\nconst CLASS_NAME_FADE$5 = 'fade';\nconst CLASS_NAME_SHOW$8 = 'show';\n\n/**\n * Class definition\n */\n\nclass Alert extends BaseComponent {\n // Getters\n static get NAME() {\n return NAME$f;\n }\n\n // Public\n close() {\n const closeEvent = EventHandler.trigger(this._element, EVENT_CLOSE);\n if (closeEvent.defaultPrevented) {\n return;\n }\n this._element.classList.remove(CLASS_NAME_SHOW$8);\n const isAnimated = this._element.classList.contains(CLASS_NAME_FADE$5);\n this._queueCallback(() => this._destroyElement(), this._element, isAnimated);\n }\n\n // Private\n _destroyElement() {\n this._element.remove();\n EventHandler.trigger(this._element, EVENT_CLOSED);\n this.dispose();\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Alert.getOrCreateInstance(this);\n if (typeof config !== 'string') {\n return;\n }\n if (data[config] === undefined || config.startsWith('_') || config === 'constructor') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config](this);\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nenableDismissTrigger(Alert, 'close');\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Alert);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap button.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$e = 'button';\nconst DATA_KEY$9 = 'bs.button';\nconst EVENT_KEY$a = `.${DATA_KEY$9}`;\nconst DATA_API_KEY$6 = '.data-api';\nconst CLASS_NAME_ACTIVE$3 = 'active';\nconst SELECTOR_DATA_TOGGLE$5 = '[data-bs-toggle=\"button\"]';\nconst EVENT_CLICK_DATA_API$6 = `click${EVENT_KEY$a}${DATA_API_KEY$6}`;\n\n/**\n * Class definition\n */\n\nclass Button extends BaseComponent {\n // Getters\n static get NAME() {\n return NAME$e;\n }\n\n // Public\n toggle() {\n // Toggle class and sync the `aria-pressed` attribute with the return value of the `.toggle()` method\n this._element.setAttribute('aria-pressed', this._element.classList.toggle(CLASS_NAME_ACTIVE$3));\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Button.getOrCreateInstance(this);\n if (config === 'toggle') {\n data[config]();\n }\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$6, SELECTOR_DATA_TOGGLE$5, event => {\n event.preventDefault();\n const button = event.target.closest(SELECTOR_DATA_TOGGLE$5);\n const data = Button.getOrCreateInstance(button);\n data.toggle();\n});\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Button);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/swipe.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$d = 'swipe';\nconst EVENT_KEY$9 = '.bs.swipe';\nconst EVENT_TOUCHSTART = `touchstart${EVENT_KEY$9}`;\nconst EVENT_TOUCHMOVE = `touchmove${EVENT_KEY$9}`;\nconst EVENT_TOUCHEND = `touchend${EVENT_KEY$9}`;\nconst EVENT_POINTERDOWN = `pointerdown${EVENT_KEY$9}`;\nconst EVENT_POINTERUP = `pointerup${EVENT_KEY$9}`;\nconst POINTER_TYPE_TOUCH = 'touch';\nconst POINTER_TYPE_PEN = 'pen';\nconst CLASS_NAME_POINTER_EVENT = 'pointer-event';\nconst SWIPE_THRESHOLD = 40;\nconst Default$c = {\n endCallback: null,\n leftCallback: null,\n rightCallback: null\n};\nconst DefaultType$c = {\n endCallback: '(function|null)',\n leftCallback: '(function|null)',\n rightCallback: '(function|null)'\n};\n\n/**\n * Class definition\n */\n\nclass Swipe extends Config {\n constructor(element, config) {\n super();\n this._element = element;\n if (!element || !Swipe.isSupported()) {\n return;\n }\n this._config = this._getConfig(config);\n this._deltaX = 0;\n this._supportPointerEvents = Boolean(window.PointerEvent);\n this._initEvents();\n }\n\n // Getters\n static get Default() {\n return Default$c;\n }\n static get DefaultType() {\n return DefaultType$c;\n }\n static get NAME() {\n return NAME$d;\n }\n\n // Public\n dispose() {\n EventHandler.off(this._element, EVENT_KEY$9);\n }\n\n // Private\n _start(event) {\n if (!this._supportPointerEvents) {\n this._deltaX = event.touches[0].clientX;\n return;\n }\n if (this._eventIsPointerPenTouch(event)) {\n this._deltaX = event.clientX;\n }\n }\n _end(event) {\n if (this._eventIsPointerPenTouch(event)) {\n this._deltaX = event.clientX - this._deltaX;\n }\n this._handleSwipe();\n execute(this._config.endCallback);\n }\n _move(event) {\n this._deltaX = event.touches && event.touches.length > 1 ? 0 : event.touches[0].clientX - this._deltaX;\n }\n _handleSwipe() {\n const absDeltaX = Math.abs(this._deltaX);\n if (absDeltaX <= SWIPE_THRESHOLD) {\n return;\n }\n const direction = absDeltaX / this._deltaX;\n this._deltaX = 0;\n if (!direction) {\n return;\n }\n execute(direction > 0 ? this._config.rightCallback : this._config.leftCallback);\n }\n _initEvents() {\n if (this._supportPointerEvents) {\n EventHandler.on(this._element, EVENT_POINTERDOWN, event => this._start(event));\n EventHandler.on(this._element, EVENT_POINTERUP, event => this._end(event));\n this._element.classList.add(CLASS_NAME_POINTER_EVENT);\n } else {\n EventHandler.on(this._element, EVENT_TOUCHSTART, event => this._start(event));\n EventHandler.on(this._element, EVENT_TOUCHMOVE, event => this._move(event));\n EventHandler.on(this._element, EVENT_TOUCHEND, event => this._end(event));\n }\n }\n _eventIsPointerPenTouch(event) {\n return this._supportPointerEvents && (event.pointerType === POINTER_TYPE_PEN || event.pointerType === POINTER_TYPE_TOUCH);\n }\n\n // Static\n static isSupported() {\n return 'ontouchstart' in document.documentElement || navigator.maxTouchPoints > 0;\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap carousel.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$c = 'carousel';\nconst DATA_KEY$8 = 'bs.carousel';\nconst EVENT_KEY$8 = `.${DATA_KEY$8}`;\nconst DATA_API_KEY$5 = '.data-api';\nconst ARROW_LEFT_KEY$1 = 'ArrowLeft';\nconst ARROW_RIGHT_KEY$1 = 'ArrowRight';\nconst TOUCHEVENT_COMPAT_WAIT = 500; // Time for mouse compat events to fire after touch\n\nconst ORDER_NEXT = 'next';\nconst ORDER_PREV = 'prev';\nconst DIRECTION_LEFT = 'left';\nconst DIRECTION_RIGHT = 'right';\nconst EVENT_SLIDE = `slide${EVENT_KEY$8}`;\nconst EVENT_SLID = `slid${EVENT_KEY$8}`;\nconst EVENT_KEYDOWN$1 = `keydown${EVENT_KEY$8}`;\nconst EVENT_MOUSEENTER$1 = `mouseenter${EVENT_KEY$8}`;\nconst EVENT_MOUSELEAVE$1 = `mouseleave${EVENT_KEY$8}`;\nconst EVENT_DRAG_START = `dragstart${EVENT_KEY$8}`;\nconst EVENT_LOAD_DATA_API$3 = `load${EVENT_KEY$8}${DATA_API_KEY$5}`;\nconst EVENT_CLICK_DATA_API$5 = `click${EVENT_KEY$8}${DATA_API_KEY$5}`;\nconst CLASS_NAME_CAROUSEL = 'carousel';\nconst CLASS_NAME_ACTIVE$2 = 'active';\nconst CLASS_NAME_SLIDE = 'slide';\nconst CLASS_NAME_END = 'carousel-item-end';\nconst CLASS_NAME_START = 'carousel-item-start';\nconst CLASS_NAME_NEXT = 'carousel-item-next';\nconst CLASS_NAME_PREV = 'carousel-item-prev';\nconst SELECTOR_ACTIVE = '.active';\nconst SELECTOR_ITEM = '.carousel-item';\nconst SELECTOR_ACTIVE_ITEM = SELECTOR_ACTIVE + SELECTOR_ITEM;\nconst SELECTOR_ITEM_IMG = '.carousel-item img';\nconst SELECTOR_INDICATORS = '.carousel-indicators';\nconst SELECTOR_DATA_SLIDE = '[data-bs-slide], [data-bs-slide-to]';\nconst SELECTOR_DATA_RIDE = '[data-bs-ride=\"carousel\"]';\nconst KEY_TO_DIRECTION = {\n [ARROW_LEFT_KEY$1]: DIRECTION_RIGHT,\n [ARROW_RIGHT_KEY$1]: DIRECTION_LEFT\n};\nconst Default$b = {\n interval: 5000,\n keyboard: true,\n pause: 'hover',\n ride: false,\n touch: true,\n wrap: true\n};\nconst DefaultType$b = {\n interval: '(number|boolean)',\n // TODO:v6 remove boolean support\n keyboard: 'boolean',\n pause: '(string|boolean)',\n ride: '(boolean|string)',\n touch: 'boolean',\n wrap: 'boolean'\n};\n\n/**\n * Class definition\n */\n\nclass Carousel extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._interval = null;\n this._activeElement = null;\n this._isSliding = false;\n this.touchTimeout = null;\n this._swipeHelper = null;\n this._indicatorsElement = SelectorEngine.findOne(SELECTOR_INDICATORS, this._element);\n this._addEventListeners();\n if (this._config.ride === CLASS_NAME_CAROUSEL) {\n this.cycle();\n }\n }\n\n // Getters\n static get Default() {\n return Default$b;\n }\n static get DefaultType() {\n return DefaultType$b;\n }\n static get NAME() {\n return NAME$c;\n }\n\n // Public\n next() {\n this._slide(ORDER_NEXT);\n }\n nextWhenVisible() {\n // FIXME TODO use `document.visibilityState`\n // Don't call next when the page isn't visible\n // or the carousel or its parent isn't visible\n if (!document.hidden && isVisible(this._element)) {\n this.next();\n }\n }\n prev() {\n this._slide(ORDER_PREV);\n }\n pause() {\n if (this._isSliding) {\n triggerTransitionEnd(this._element);\n }\n this._clearInterval();\n }\n cycle() {\n this._clearInterval();\n this._updateInterval();\n this._interval = setInterval(() => this.nextWhenVisible(), this._config.interval);\n }\n _maybeEnableCycle() {\n if (!this._config.ride) {\n return;\n }\n if (this._isSliding) {\n EventHandler.one(this._element, EVENT_SLID, () => this.cycle());\n return;\n }\n this.cycle();\n }\n to(index) {\n const items = this._getItems();\n if (index > items.length - 1 || index < 0) {\n return;\n }\n if (this._isSliding) {\n EventHandler.one(this._element, EVENT_SLID, () => this.to(index));\n return;\n }\n const activeIndex = this._getItemIndex(this._getActive());\n if (activeIndex === index) {\n return;\n }\n const order = index > activeIndex ? ORDER_NEXT : ORDER_PREV;\n this._slide(order, items[index]);\n }\n dispose() {\n if (this._swipeHelper) {\n this._swipeHelper.dispose();\n }\n super.dispose();\n }\n\n // Private\n _configAfterMerge(config) {\n config.defaultInterval = config.interval;\n return config;\n }\n _addEventListeners() {\n if (this._config.keyboard) {\n EventHandler.on(this._element, EVENT_KEYDOWN$1, event => this._keydown(event));\n }\n if (this._config.pause === 'hover') {\n EventHandler.on(this._element, EVENT_MOUSEENTER$1, () => this.pause());\n EventHandler.on(this._element, EVENT_MOUSELEAVE$1, () => this._maybeEnableCycle());\n }\n if (this._config.touch && Swipe.isSupported()) {\n this._addTouchEventListeners();\n }\n }\n _addTouchEventListeners() {\n for (const img of SelectorEngine.find(SELECTOR_ITEM_IMG, this._element)) {\n EventHandler.on(img, EVENT_DRAG_START, event => event.preventDefault());\n }\n const endCallBack = () => {\n if (this._config.pause !== 'hover') {\n return;\n }\n\n // If it's a touch-enabled device, mouseenter/leave are fired as\n // part of the mouse compatibility events on first tap - the carousel\n // would stop cycling until user tapped out of it;\n // here, we listen for touchend, explicitly pause the carousel\n // (as if it's the second time we tap on it, mouseenter compat event\n // is NOT fired) and after a timeout (to allow for mouse compatibility\n // events to fire) we explicitly restart cycling\n\n this.pause();\n if (this.touchTimeout) {\n clearTimeout(this.touchTimeout);\n }\n this.touchTimeout = setTimeout(() => this._maybeEnableCycle(), TOUCHEVENT_COMPAT_WAIT + this._config.interval);\n };\n const swipeConfig = {\n leftCallback: () => this._slide(this._directionToOrder(DIRECTION_LEFT)),\n rightCallback: () => this._slide(this._directionToOrder(DIRECTION_RIGHT)),\n endCallback: endCallBack\n };\n this._swipeHelper = new Swipe(this._element, swipeConfig);\n }\n _keydown(event) {\n if (/input|textarea/i.test(event.target.tagName)) {\n return;\n }\n const direction = KEY_TO_DIRECTION[event.key];\n if (direction) {\n event.preventDefault();\n this._slide(this._directionToOrder(direction));\n }\n }\n _getItemIndex(element) {\n return this._getItems().indexOf(element);\n }\n _setActiveIndicatorElement(index) {\n if (!this._indicatorsElement) {\n return;\n }\n const activeIndicator = SelectorEngine.findOne(SELECTOR_ACTIVE, this._indicatorsElement);\n activeIndicator.classList.remove(CLASS_NAME_ACTIVE$2);\n activeIndicator.removeAttribute('aria-current');\n const newActiveIndicator = SelectorEngine.findOne(`[data-bs-slide-to=\"${index}\"]`, this._indicatorsElement);\n if (newActiveIndicator) {\n newActiveIndicator.classList.add(CLASS_NAME_ACTIVE$2);\n newActiveIndicator.setAttribute('aria-current', 'true');\n }\n }\n _updateInterval() {\n const element = this._activeElement || this._getActive();\n if (!element) {\n return;\n }\n const elementInterval = Number.parseInt(element.getAttribute('data-bs-interval'), 10);\n this._config.interval = elementInterval || this._config.defaultInterval;\n }\n _slide(order, element = null) {\n if (this._isSliding) {\n return;\n }\n const activeElement = this._getActive();\n const isNext = order === ORDER_NEXT;\n const nextElement = element || getNextActiveElement(this._getItems(), activeElement, isNext, this._config.wrap);\n if (nextElement === activeElement) {\n return;\n }\n const nextElementIndex = this._getItemIndex(nextElement);\n const triggerEvent = eventName => {\n return EventHandler.trigger(this._element, eventName, {\n relatedTarget: nextElement,\n direction: this._orderToDirection(order),\n from: this._getItemIndex(activeElement),\n to: nextElementIndex\n });\n };\n const slideEvent = triggerEvent(EVENT_SLIDE);\n if (slideEvent.defaultPrevented) {\n return;\n }\n if (!activeElement || !nextElement) {\n // Some weirdness is happening, so we bail\n // TODO: change tests that use empty divs to avoid this check\n return;\n }\n const isCycling = Boolean(this._interval);\n this.pause();\n this._isSliding = true;\n this._setActiveIndicatorElement(nextElementIndex);\n this._activeElement = nextElement;\n const directionalClassName = isNext ? CLASS_NAME_START : CLASS_NAME_END;\n const orderClassName = isNext ? CLASS_NAME_NEXT : CLASS_NAME_PREV;\n nextElement.classList.add(orderClassName);\n reflow(nextElement);\n activeElement.classList.add(directionalClassName);\n nextElement.classList.add(directionalClassName);\n const completeCallBack = () => {\n nextElement.classList.remove(directionalClassName, orderClassName);\n nextElement.classList.add(CLASS_NAME_ACTIVE$2);\n activeElement.classList.remove(CLASS_NAME_ACTIVE$2, orderClassName, directionalClassName);\n this._isSliding = false;\n triggerEvent(EVENT_SLID);\n };\n this._queueCallback(completeCallBack, activeElement, this._isAnimated());\n if (isCycling) {\n this.cycle();\n }\n }\n _isAnimated() {\n return this._element.classList.contains(CLASS_NAME_SLIDE);\n }\n _getActive() {\n return SelectorEngine.findOne(SELECTOR_ACTIVE_ITEM, this._element);\n }\n _getItems() {\n return SelectorEngine.find(SELECTOR_ITEM, this._element);\n }\n _clearInterval() {\n if (this._interval) {\n clearInterval(this._interval);\n this._interval = null;\n }\n }\n _directionToOrder(direction) {\n if (isRTL()) {\n return direction === DIRECTION_LEFT ? ORDER_PREV : ORDER_NEXT;\n }\n return direction === DIRECTION_LEFT ? ORDER_NEXT : ORDER_PREV;\n }\n _orderToDirection(order) {\n if (isRTL()) {\n return order === ORDER_PREV ? DIRECTION_LEFT : DIRECTION_RIGHT;\n }\n return order === ORDER_PREV ? DIRECTION_RIGHT : DIRECTION_LEFT;\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Carousel.getOrCreateInstance(this, config);\n if (typeof config === 'number') {\n data.to(config);\n return;\n }\n if (typeof config === 'string') {\n if (data[config] === undefined || config.startsWith('_') || config === 'constructor') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n }\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$5, SELECTOR_DATA_SLIDE, function (event) {\n const target = SelectorEngine.getElementFromSelector(this);\n if (!target || !target.classList.contains(CLASS_NAME_CAROUSEL)) {\n return;\n }\n event.preventDefault();\n const carousel = Carousel.getOrCreateInstance(target);\n const slideIndex = this.getAttribute('data-bs-slide-to');\n if (slideIndex) {\n carousel.to(slideIndex);\n carousel._maybeEnableCycle();\n return;\n }\n if (Manipulator.getDataAttribute(this, 'slide') === 'next') {\n carousel.next();\n carousel._maybeEnableCycle();\n return;\n }\n carousel.prev();\n carousel._maybeEnableCycle();\n});\nEventHandler.on(window, EVENT_LOAD_DATA_API$3, () => {\n const carousels = SelectorEngine.find(SELECTOR_DATA_RIDE);\n for (const carousel of carousels) {\n Carousel.getOrCreateInstance(carousel);\n }\n});\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Carousel);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap collapse.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$b = 'collapse';\nconst DATA_KEY$7 = 'bs.collapse';\nconst EVENT_KEY$7 = `.${DATA_KEY$7}`;\nconst DATA_API_KEY$4 = '.data-api';\nconst EVENT_SHOW$6 = `show${EVENT_KEY$7}`;\nconst EVENT_SHOWN$6 = `shown${EVENT_KEY$7}`;\nconst EVENT_HIDE$6 = `hide${EVENT_KEY$7}`;\nconst EVENT_HIDDEN$6 = `hidden${EVENT_KEY$7}`;\nconst EVENT_CLICK_DATA_API$4 = `click${EVENT_KEY$7}${DATA_API_KEY$4}`;\nconst CLASS_NAME_SHOW$7 = 'show';\nconst CLASS_NAME_COLLAPSE = 'collapse';\nconst CLASS_NAME_COLLAPSING = 'collapsing';\nconst CLASS_NAME_COLLAPSED = 'collapsed';\nconst CLASS_NAME_DEEPER_CHILDREN = `:scope .${CLASS_NAME_COLLAPSE} .${CLASS_NAME_COLLAPSE}`;\nconst CLASS_NAME_HORIZONTAL = 'collapse-horizontal';\nconst WIDTH = 'width';\nconst HEIGHT = 'height';\nconst SELECTOR_ACTIVES = '.collapse.show, .collapse.collapsing';\nconst SELECTOR_DATA_TOGGLE$4 = '[data-bs-toggle=\"collapse\"]';\nconst Default$a = {\n parent: null,\n toggle: true\n};\nconst DefaultType$a = {\n parent: '(null|element)',\n toggle: 'boolean'\n};\n\n/**\n * Class definition\n */\n\nclass Collapse extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._isTransitioning = false;\n this._triggerArray = [];\n const toggleList = SelectorEngine.find(SELECTOR_DATA_TOGGLE$4);\n for (const elem of toggleList) {\n const selector = SelectorEngine.getSelectorFromElement(elem);\n const filterElement = SelectorEngine.find(selector).filter(foundElement => foundElement === this._element);\n if (selector !== null && filterElement.length) {\n this._triggerArray.push(elem);\n }\n }\n this._initializeChildren();\n if (!this._config.parent) {\n this._addAriaAndCollapsedClass(this._triggerArray, this._isShown());\n }\n if (this._config.toggle) {\n this.toggle();\n }\n }\n\n // Getters\n static get Default() {\n return Default$a;\n }\n static get DefaultType() {\n return DefaultType$a;\n }\n static get NAME() {\n return NAME$b;\n }\n\n // Public\n toggle() {\n if (this._isShown()) {\n this.hide();\n } else {\n this.show();\n }\n }\n show() {\n if (this._isTransitioning || this._isShown()) {\n return;\n }\n let activeChildren = [];\n\n // find active children\n if (this._config.parent) {\n activeChildren = this._getFirstLevelChildren(SELECTOR_ACTIVES).filter(element => element !== this._element).map(element => Collapse.getOrCreateInstance(element, {\n toggle: false\n }));\n }\n if (activeChildren.length && activeChildren[0]._isTransitioning) {\n return;\n }\n const startEvent = EventHandler.trigger(this._element, EVENT_SHOW$6);\n if (startEvent.defaultPrevented) {\n return;\n }\n for (const activeInstance of activeChildren) {\n activeInstance.hide();\n }\n const dimension = this._getDimension();\n this._element.classList.remove(CLASS_NAME_COLLAPSE);\n this._element.classList.add(CLASS_NAME_COLLAPSING);\n this._element.style[dimension] = 0;\n this._addAriaAndCollapsedClass(this._triggerArray, true);\n this._isTransitioning = true;\n const complete = () => {\n this._isTransitioning = false;\n this._element.classList.remove(CLASS_NAME_COLLAPSING);\n this._element.classList.add(CLASS_NAME_COLLAPSE, CLASS_NAME_SHOW$7);\n this._element.style[dimension] = '';\n EventHandler.trigger(this._element, EVENT_SHOWN$6);\n };\n const capitalizedDimension = dimension[0].toUpperCase() + dimension.slice(1);\n const scrollSize = `scroll${capitalizedDimension}`;\n this._queueCallback(complete, this._element, true);\n this._element.style[dimension] = `${this._element[scrollSize]}px`;\n }\n hide() {\n if (this._isTransitioning || !this._isShown()) {\n return;\n }\n const startEvent = EventHandler.trigger(this._element, EVENT_HIDE$6);\n if (startEvent.defaultPrevented) {\n return;\n }\n const dimension = this._getDimension();\n this._element.style[dimension] = `${this._element.getBoundingClientRect()[dimension]}px`;\n reflow(this._element);\n this._element.classList.add(CLASS_NAME_COLLAPSING);\n this._element.classList.remove(CLASS_NAME_COLLAPSE, CLASS_NAME_SHOW$7);\n for (const trigger of this._triggerArray) {\n const element = SelectorEngine.getElementFromSelector(trigger);\n if (element && !this._isShown(element)) {\n this._addAriaAndCollapsedClass([trigger], false);\n }\n }\n this._isTransitioning = true;\n const complete = () => {\n this._isTransitioning = false;\n this._element.classList.remove(CLASS_NAME_COLLAPSING);\n this._element.classList.add(CLASS_NAME_COLLAPSE);\n EventHandler.trigger(this._element, EVENT_HIDDEN$6);\n };\n this._element.style[dimension] = '';\n this._queueCallback(complete, this._element, true);\n }\n _isShown(element = this._element) {\n return element.classList.contains(CLASS_NAME_SHOW$7);\n }\n\n // Private\n _configAfterMerge(config) {\n config.toggle = Boolean(config.toggle); // Coerce string values\n config.parent = getElement(config.parent);\n return config;\n }\n _getDimension() {\n return this._element.classList.contains(CLASS_NAME_HORIZONTAL) ? WIDTH : HEIGHT;\n }\n _initializeChildren() {\n if (!this._config.parent) {\n return;\n }\n const children = this._getFirstLevelChildren(SELECTOR_DATA_TOGGLE$4);\n for (const element of children) {\n const selected = SelectorEngine.getElementFromSelector(element);\n if (selected) {\n this._addAriaAndCollapsedClass([element], this._isShown(selected));\n }\n }\n }\n _getFirstLevelChildren(selector) {\n const children = SelectorEngine.find(CLASS_NAME_DEEPER_CHILDREN, this._config.parent);\n // remove children if greater depth\n return SelectorEngine.find(selector, this._config.parent).filter(element => !children.includes(element));\n }\n _addAriaAndCollapsedClass(triggerArray, isOpen) {\n if (!triggerArray.length) {\n return;\n }\n for (const element of triggerArray) {\n element.classList.toggle(CLASS_NAME_COLLAPSED, !isOpen);\n element.setAttribute('aria-expanded', isOpen);\n }\n }\n\n // Static\n static jQueryInterface(config) {\n const _config = {};\n if (typeof config === 'string' && /show|hide/.test(config)) {\n _config.toggle = false;\n }\n return this.each(function () {\n const data = Collapse.getOrCreateInstance(this, _config);\n if (typeof config === 'string') {\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n }\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$4, SELECTOR_DATA_TOGGLE$4, function (event) {\n // preventDefault only for elements (which change the URL) not inside the collapsible element\n if (event.target.tagName === 'A' || event.delegateTarget && event.delegateTarget.tagName === 'A') {\n event.preventDefault();\n }\n for (const element of SelectorEngine.getMultipleElementsFromSelector(this)) {\n Collapse.getOrCreateInstance(element, {\n toggle: false\n }).toggle();\n }\n});\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Collapse);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dropdown.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$a = 'dropdown';\nconst DATA_KEY$6 = 'bs.dropdown';\nconst EVENT_KEY$6 = `.${DATA_KEY$6}`;\nconst DATA_API_KEY$3 = '.data-api';\nconst ESCAPE_KEY$2 = 'Escape';\nconst TAB_KEY$1 = 'Tab';\nconst ARROW_UP_KEY$1 = 'ArrowUp';\nconst ARROW_DOWN_KEY$1 = 'ArrowDown';\nconst RIGHT_MOUSE_BUTTON = 2; // MouseEvent.button value for the secondary button, usually the right button\n\nconst EVENT_HIDE$5 = `hide${EVENT_KEY$6}`;\nconst EVENT_HIDDEN$5 = `hidden${EVENT_KEY$6}`;\nconst EVENT_SHOW$5 = `show${EVENT_KEY$6}`;\nconst EVENT_SHOWN$5 = `shown${EVENT_KEY$6}`;\nconst EVENT_CLICK_DATA_API$3 = `click${EVENT_KEY$6}${DATA_API_KEY$3}`;\nconst EVENT_KEYDOWN_DATA_API = `keydown${EVENT_KEY$6}${DATA_API_KEY$3}`;\nconst EVENT_KEYUP_DATA_API = `keyup${EVENT_KEY$6}${DATA_API_KEY$3}`;\nconst CLASS_NAME_SHOW$6 = 'show';\nconst CLASS_NAME_DROPUP = 'dropup';\nconst CLASS_NAME_DROPEND = 'dropend';\nconst CLASS_NAME_DROPSTART = 'dropstart';\nconst CLASS_NAME_DROPUP_CENTER = 'dropup-center';\nconst CLASS_NAME_DROPDOWN_CENTER = 'dropdown-center';\nconst SELECTOR_DATA_TOGGLE$3 = '[data-bs-toggle=\"dropdown\"]:not(.disabled):not(:disabled)';\nconst SELECTOR_DATA_TOGGLE_SHOWN = `${SELECTOR_DATA_TOGGLE$3}.${CLASS_NAME_SHOW$6}`;\nconst SELECTOR_MENU = '.dropdown-menu';\nconst SELECTOR_NAVBAR = '.navbar';\nconst SELECTOR_NAVBAR_NAV = '.navbar-nav';\nconst SELECTOR_VISIBLE_ITEMS = '.dropdown-menu .dropdown-item:not(.disabled):not(:disabled)';\nconst PLACEMENT_TOP = isRTL() ? 'top-end' : 'top-start';\nconst PLACEMENT_TOPEND = isRTL() ? 'top-start' : 'top-end';\nconst PLACEMENT_BOTTOM = isRTL() ? 'bottom-end' : 'bottom-start';\nconst PLACEMENT_BOTTOMEND = isRTL() ? 'bottom-start' : 'bottom-end';\nconst PLACEMENT_RIGHT = isRTL() ? 'left-start' : 'right-start';\nconst PLACEMENT_LEFT = isRTL() ? 'right-start' : 'left-start';\nconst PLACEMENT_TOPCENTER = 'top';\nconst PLACEMENT_BOTTOMCENTER = 'bottom';\nconst Default$9 = {\n autoClose: true,\n boundary: 'clippingParents',\n display: 'dynamic',\n offset: [0, 2],\n popperConfig: null,\n reference: 'toggle'\n};\nconst DefaultType$9 = {\n autoClose: '(boolean|string)',\n boundary: '(string|element)',\n display: 'string',\n offset: '(array|string|function)',\n popperConfig: '(null|object|function)',\n reference: '(string|element|object)'\n};\n\n/**\n * Class definition\n */\n\nclass Dropdown extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._popper = null;\n this._parent = this._element.parentNode; // dropdown wrapper\n // TODO: v6 revert #37011 & change markup https://getbootstrap.com/docs/5.3/forms/input-group/\n this._menu = SelectorEngine.next(this._element, SELECTOR_MENU)[0] || SelectorEngine.prev(this._element, SELECTOR_MENU)[0] || SelectorEngine.findOne(SELECTOR_MENU, this._parent);\n this._inNavbar = this._detectNavbar();\n }\n\n // Getters\n static get Default() {\n return Default$9;\n }\n static get DefaultType() {\n return DefaultType$9;\n }\n static get NAME() {\n return NAME$a;\n }\n\n // Public\n toggle() {\n return this._isShown() ? this.hide() : this.show();\n }\n show() {\n if (isDisabled(this._element) || this._isShown()) {\n return;\n }\n const relatedTarget = {\n relatedTarget: this._element\n };\n const showEvent = EventHandler.trigger(this._element, EVENT_SHOW$5, relatedTarget);\n if (showEvent.defaultPrevented) {\n return;\n }\n this._createPopper();\n\n // If this is a touch-enabled device we add extra\n // empty mouseover listeners to the body's immediate children;\n // only needed because of broken event delegation on iOS\n // https://www.quirksmode.org/blog/archives/2014/02/mouse_event_bub.html\n if ('ontouchstart' in document.documentElement && !this._parent.closest(SELECTOR_NAVBAR_NAV)) {\n for (const element of [].concat(...document.body.children)) {\n EventHandler.on(element, 'mouseover', noop);\n }\n }\n this._element.focus();\n this._element.setAttribute('aria-expanded', true);\n this._menu.classList.add(CLASS_NAME_SHOW$6);\n this._element.classList.add(CLASS_NAME_SHOW$6);\n EventHandler.trigger(this._element, EVENT_SHOWN$5, relatedTarget);\n }\n hide() {\n if (isDisabled(this._element) || !this._isShown()) {\n return;\n }\n const relatedTarget = {\n relatedTarget: this._element\n };\n this._completeHide(relatedTarget);\n }\n dispose() {\n if (this._popper) {\n this._popper.destroy();\n }\n super.dispose();\n }\n update() {\n this._inNavbar = this._detectNavbar();\n if (this._popper) {\n this._popper.update();\n }\n }\n\n // Private\n _completeHide(relatedTarget) {\n const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE$5, relatedTarget);\n if (hideEvent.defaultPrevented) {\n return;\n }\n\n // If this is a touch-enabled device we remove the extra\n // empty mouseover listeners we added for iOS support\n if ('ontouchstart' in document.documentElement) {\n for (const element of [].concat(...document.body.children)) {\n EventHandler.off(element, 'mouseover', noop);\n }\n }\n if (this._popper) {\n this._popper.destroy();\n }\n this._menu.classList.remove(CLASS_NAME_SHOW$6);\n this._element.classList.remove(CLASS_NAME_SHOW$6);\n this._element.setAttribute('aria-expanded', 'false');\n Manipulator.removeDataAttribute(this._menu, 'popper');\n EventHandler.trigger(this._element, EVENT_HIDDEN$5, relatedTarget);\n }\n _getConfig(config) {\n config = super._getConfig(config);\n if (typeof config.reference === 'object' && !isElement(config.reference) && typeof config.reference.getBoundingClientRect !== 'function') {\n // Popper virtual elements require a getBoundingClientRect method\n throw new TypeError(`${NAME$a.toUpperCase()}: Option \"reference\" provided type \"object\" without a required \"getBoundingClientRect\" method.`);\n }\n return config;\n }\n _createPopper() {\n if (typeof Popper === 'undefined') {\n throw new TypeError('Bootstrap\\'s dropdowns require Popper (https://popper.js.org)');\n }\n let referenceElement = this._element;\n if (this._config.reference === 'parent') {\n referenceElement = this._parent;\n } else if (isElement(this._config.reference)) {\n referenceElement = getElement(this._config.reference);\n } else if (typeof this._config.reference === 'object') {\n referenceElement = this._config.reference;\n }\n const popperConfig = this._getPopperConfig();\n this._popper = Popper.createPopper(referenceElement, this._menu, popperConfig);\n }\n _isShown() {\n return this._menu.classList.contains(CLASS_NAME_SHOW$6);\n }\n _getPlacement() {\n const parentDropdown = this._parent;\n if (parentDropdown.classList.contains(CLASS_NAME_DROPEND)) {\n return PLACEMENT_RIGHT;\n }\n if (parentDropdown.classList.contains(CLASS_NAME_DROPSTART)) {\n return PLACEMENT_LEFT;\n }\n if (parentDropdown.classList.contains(CLASS_NAME_DROPUP_CENTER)) {\n return PLACEMENT_TOPCENTER;\n }\n if (parentDropdown.classList.contains(CLASS_NAME_DROPDOWN_CENTER)) {\n return PLACEMENT_BOTTOMCENTER;\n }\n\n // We need to trim the value because custom properties can also include spaces\n const isEnd = getComputedStyle(this._menu).getPropertyValue('--bs-position').trim() === 'end';\n if (parentDropdown.classList.contains(CLASS_NAME_DROPUP)) {\n return isEnd ? PLACEMENT_TOPEND : PLACEMENT_TOP;\n }\n return isEnd ? PLACEMENT_BOTTOMEND : PLACEMENT_BOTTOM;\n }\n _detectNavbar() {\n return this._element.closest(SELECTOR_NAVBAR) !== null;\n }\n _getOffset() {\n const {\n offset\n } = this._config;\n if (typeof offset === 'string') {\n return offset.split(',').map(value => Number.parseInt(value, 10));\n }\n if (typeof offset === 'function') {\n return popperData => offset(popperData, this._element);\n }\n return offset;\n }\n _getPopperConfig() {\n const defaultBsPopperConfig = {\n placement: this._getPlacement(),\n modifiers: [{\n name: 'preventOverflow',\n options: {\n boundary: this._config.boundary\n }\n }, {\n name: 'offset',\n options: {\n offset: this._getOffset()\n }\n }]\n };\n\n // Disable Popper if we have a static display or Dropdown is in Navbar\n if (this._inNavbar || this._config.display === 'static') {\n Manipulator.setDataAttribute(this._menu, 'popper', 'static'); // TODO: v6 remove\n defaultBsPopperConfig.modifiers = [{\n name: 'applyStyles',\n enabled: false\n }];\n }\n return {\n ...defaultBsPopperConfig,\n ...execute(this._config.popperConfig, [defaultBsPopperConfig])\n };\n }\n _selectMenuItem({\n key,\n target\n }) {\n const items = SelectorEngine.find(SELECTOR_VISIBLE_ITEMS, this._menu).filter(element => isVisible(element));\n if (!items.length) {\n return;\n }\n\n // if target isn't included in items (e.g. when expanding the dropdown)\n // allow cycling to get the last item in case key equals ARROW_UP_KEY\n getNextActiveElement(items, target, key === ARROW_DOWN_KEY$1, !items.includes(target)).focus();\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Dropdown.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n });\n }\n static clearMenus(event) {\n if (event.button === RIGHT_MOUSE_BUTTON || event.type === 'keyup' && event.key !== TAB_KEY$1) {\n return;\n }\n const openToggles = SelectorEngine.find(SELECTOR_DATA_TOGGLE_SHOWN);\n for (const toggle of openToggles) {\n const context = Dropdown.getInstance(toggle);\n if (!context || context._config.autoClose === false) {\n continue;\n }\n const composedPath = event.composedPath();\n const isMenuTarget = composedPath.includes(context._menu);\n if (composedPath.includes(context._element) || context._config.autoClose === 'inside' && !isMenuTarget || context._config.autoClose === 'outside' && isMenuTarget) {\n continue;\n }\n\n // Tab navigation through the dropdown menu or events from contained inputs shouldn't close the menu\n if (context._menu.contains(event.target) && (event.type === 'keyup' && event.key === TAB_KEY$1 || /input|select|option|textarea|form/i.test(event.target.tagName))) {\n continue;\n }\n const relatedTarget = {\n relatedTarget: context._element\n };\n if (event.type === 'click') {\n relatedTarget.clickEvent = event;\n }\n context._completeHide(relatedTarget);\n }\n }\n static dataApiKeydownHandler(event) {\n // If not an UP | DOWN | ESCAPE key => not a dropdown command\n // If input/textarea && if key is other than ESCAPE => not a dropdown command\n\n const isInput = /input|textarea/i.test(event.target.tagName);\n const isEscapeEvent = event.key === ESCAPE_KEY$2;\n const isUpOrDownEvent = [ARROW_UP_KEY$1, ARROW_DOWN_KEY$1].includes(event.key);\n if (!isUpOrDownEvent && !isEscapeEvent) {\n return;\n }\n if (isInput && !isEscapeEvent) {\n return;\n }\n event.preventDefault();\n\n // TODO: v6 revert #37011 & change markup https://getbootstrap.com/docs/5.3/forms/input-group/\n const getToggleButton = this.matches(SELECTOR_DATA_TOGGLE$3) ? this : SelectorEngine.prev(this, SELECTOR_DATA_TOGGLE$3)[0] || SelectorEngine.next(this, SELECTOR_DATA_TOGGLE$3)[0] || SelectorEngine.findOne(SELECTOR_DATA_TOGGLE$3, event.delegateTarget.parentNode);\n const instance = Dropdown.getOrCreateInstance(getToggleButton);\n if (isUpOrDownEvent) {\n event.stopPropagation();\n instance.show();\n instance._selectMenuItem(event);\n return;\n }\n if (instance._isShown()) {\n // else is escape and we check if it is shown\n event.stopPropagation();\n instance.hide();\n getToggleButton.focus();\n }\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_KEYDOWN_DATA_API, SELECTOR_DATA_TOGGLE$3, Dropdown.dataApiKeydownHandler);\nEventHandler.on(document, EVENT_KEYDOWN_DATA_API, SELECTOR_MENU, Dropdown.dataApiKeydownHandler);\nEventHandler.on(document, EVENT_CLICK_DATA_API$3, Dropdown.clearMenus);\nEventHandler.on(document, EVENT_KEYUP_DATA_API, Dropdown.clearMenus);\nEventHandler.on(document, EVENT_CLICK_DATA_API$3, SELECTOR_DATA_TOGGLE$3, function (event) {\n event.preventDefault();\n Dropdown.getOrCreateInstance(this).toggle();\n});\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Dropdown);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/backdrop.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$9 = 'backdrop';\nconst CLASS_NAME_FADE$4 = 'fade';\nconst CLASS_NAME_SHOW$5 = 'show';\nconst EVENT_MOUSEDOWN = `mousedown.bs.${NAME$9}`;\nconst Default$8 = {\n className: 'modal-backdrop',\n clickCallback: null,\n isAnimated: false,\n isVisible: true,\n // if false, we use the backdrop helper without adding any element to the dom\n rootElement: 'body' // give the choice to place backdrop under different elements\n};\nconst DefaultType$8 = {\n className: 'string',\n clickCallback: '(function|null)',\n isAnimated: 'boolean',\n isVisible: 'boolean',\n rootElement: '(element|string)'\n};\n\n/**\n * Class definition\n */\n\nclass Backdrop extends Config {\n constructor(config) {\n super();\n this._config = this._getConfig(config);\n this._isAppended = false;\n this._element = null;\n }\n\n // Getters\n static get Default() {\n return Default$8;\n }\n static get DefaultType() {\n return DefaultType$8;\n }\n static get NAME() {\n return NAME$9;\n }\n\n // Public\n show(callback) {\n if (!this._config.isVisible) {\n execute(callback);\n return;\n }\n this._append();\n const element = this._getElement();\n if (this._config.isAnimated) {\n reflow(element);\n }\n element.classList.add(CLASS_NAME_SHOW$5);\n this._emulateAnimation(() => {\n execute(callback);\n });\n }\n hide(callback) {\n if (!this._config.isVisible) {\n execute(callback);\n return;\n }\n this._getElement().classList.remove(CLASS_NAME_SHOW$5);\n this._emulateAnimation(() => {\n this.dispose();\n execute(callback);\n });\n }\n dispose() {\n if (!this._isAppended) {\n return;\n }\n EventHandler.off(this._element, EVENT_MOUSEDOWN);\n this._element.remove();\n this._isAppended = false;\n }\n\n // Private\n _getElement() {\n if (!this._element) {\n const backdrop = document.createElement('div');\n backdrop.className = this._config.className;\n if (this._config.isAnimated) {\n backdrop.classList.add(CLASS_NAME_FADE$4);\n }\n this._element = backdrop;\n }\n return this._element;\n }\n _configAfterMerge(config) {\n // use getElement() with the default \"body\" to get a fresh Element on each instantiation\n config.rootElement = getElement(config.rootElement);\n return config;\n }\n _append() {\n if (this._isAppended) {\n return;\n }\n const element = this._getElement();\n this._config.rootElement.append(element);\n EventHandler.on(element, EVENT_MOUSEDOWN, () => {\n execute(this._config.clickCallback);\n });\n this._isAppended = true;\n }\n _emulateAnimation(callback) {\n executeAfterTransition(callback, this._getElement(), this._config.isAnimated);\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/focustrap.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$8 = 'focustrap';\nconst DATA_KEY$5 = 'bs.focustrap';\nconst EVENT_KEY$5 = `.${DATA_KEY$5}`;\nconst EVENT_FOCUSIN$2 = `focusin${EVENT_KEY$5}`;\nconst EVENT_KEYDOWN_TAB = `keydown.tab${EVENT_KEY$5}`;\nconst TAB_KEY = 'Tab';\nconst TAB_NAV_FORWARD = 'forward';\nconst TAB_NAV_BACKWARD = 'backward';\nconst Default$7 = {\n autofocus: true,\n trapElement: null // The element to trap focus inside of\n};\nconst DefaultType$7 = {\n autofocus: 'boolean',\n trapElement: 'element'\n};\n\n/**\n * Class definition\n */\n\nclass FocusTrap extends Config {\n constructor(config) {\n super();\n this._config = this._getConfig(config);\n this._isActive = false;\n this._lastTabNavDirection = null;\n }\n\n // Getters\n static get Default() {\n return Default$7;\n }\n static get DefaultType() {\n return DefaultType$7;\n }\n static get NAME() {\n return NAME$8;\n }\n\n // Public\n activate() {\n if (this._isActive) {\n return;\n }\n if (this._config.autofocus) {\n this._config.trapElement.focus();\n }\n EventHandler.off(document, EVENT_KEY$5); // guard against infinite focus loop\n EventHandler.on(document, EVENT_FOCUSIN$2, event => this._handleFocusin(event));\n EventHandler.on(document, EVENT_KEYDOWN_TAB, event => this._handleKeydown(event));\n this._isActive = true;\n }\n deactivate() {\n if (!this._isActive) {\n return;\n }\n this._isActive = false;\n EventHandler.off(document, EVENT_KEY$5);\n }\n\n // Private\n _handleFocusin(event) {\n const {\n trapElement\n } = this._config;\n if (event.target === document || event.target === trapElement || trapElement.contains(event.target)) {\n return;\n }\n const elements = SelectorEngine.focusableChildren(trapElement);\n if (elements.length === 0) {\n trapElement.focus();\n } else if (this._lastTabNavDirection === TAB_NAV_BACKWARD) {\n elements[elements.length - 1].focus();\n } else {\n elements[0].focus();\n }\n }\n _handleKeydown(event) {\n if (event.key !== TAB_KEY) {\n return;\n }\n this._lastTabNavDirection = event.shiftKey ? TAB_NAV_BACKWARD : TAB_NAV_FORWARD;\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/scrollBar.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst SELECTOR_FIXED_CONTENT = '.fixed-top, .fixed-bottom, .is-fixed, .sticky-top';\nconst SELECTOR_STICKY_CONTENT = '.sticky-top';\nconst PROPERTY_PADDING = 'padding-right';\nconst PROPERTY_MARGIN = 'margin-right';\n\n/**\n * Class definition\n */\n\nclass ScrollBarHelper {\n constructor() {\n this._element = document.body;\n }\n\n // Public\n getWidth() {\n // https://developer.mozilla.org/en-US/docs/Web/API/Window/innerWidth#usage_notes\n const documentWidth = document.documentElement.clientWidth;\n return Math.abs(window.innerWidth - documentWidth);\n }\n hide() {\n const width = this.getWidth();\n this._disableOverFlow();\n // give padding to element to balance the hidden scrollbar width\n this._setElementAttributes(this._element, PROPERTY_PADDING, calculatedValue => calculatedValue + width);\n // trick: We adjust positive paddingRight and negative marginRight to sticky-top elements to keep showing fullwidth\n this._setElementAttributes(SELECTOR_FIXED_CONTENT, PROPERTY_PADDING, calculatedValue => calculatedValue + width);\n this._setElementAttributes(SELECTOR_STICKY_CONTENT, PROPERTY_MARGIN, calculatedValue => calculatedValue - width);\n }\n reset() {\n this._resetElementAttributes(this._element, 'overflow');\n this._resetElementAttributes(this._element, PROPERTY_PADDING);\n this._resetElementAttributes(SELECTOR_FIXED_CONTENT, PROPERTY_PADDING);\n this._resetElementAttributes(SELECTOR_STICKY_CONTENT, PROPERTY_MARGIN);\n }\n isOverflowing() {\n return this.getWidth() > 0;\n }\n\n // Private\n _disableOverFlow() {\n this._saveInitialAttribute(this._element, 'overflow');\n this._element.style.overflow = 'hidden';\n }\n _setElementAttributes(selector, styleProperty, callback) {\n const scrollbarWidth = this.getWidth();\n const manipulationCallBack = element => {\n if (element !== this._element && window.innerWidth > element.clientWidth + scrollbarWidth) {\n return;\n }\n this._saveInitialAttribute(element, styleProperty);\n const calculatedValue = window.getComputedStyle(element).getPropertyValue(styleProperty);\n element.style.setProperty(styleProperty, `${callback(Number.parseFloat(calculatedValue))}px`);\n };\n this._applyManipulationCallback(selector, manipulationCallBack);\n }\n _saveInitialAttribute(element, styleProperty) {\n const actualValue = element.style.getPropertyValue(styleProperty);\n if (actualValue) {\n Manipulator.setDataAttribute(element, styleProperty, actualValue);\n }\n }\n _resetElementAttributes(selector, styleProperty) {\n const manipulationCallBack = element => {\n const value = Manipulator.getDataAttribute(element, styleProperty);\n // We only want to remove the property if the value is `null`; the value can also be zero\n if (value === null) {\n element.style.removeProperty(styleProperty);\n return;\n }\n Manipulator.removeDataAttribute(element, styleProperty);\n element.style.setProperty(styleProperty, value);\n };\n this._applyManipulationCallback(selector, manipulationCallBack);\n }\n _applyManipulationCallback(selector, callBack) {\n if (isElement(selector)) {\n callBack(selector);\n return;\n }\n for (const sel of SelectorEngine.find(selector, this._element)) {\n callBack(sel);\n }\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap modal.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$7 = 'modal';\nconst DATA_KEY$4 = 'bs.modal';\nconst EVENT_KEY$4 = `.${DATA_KEY$4}`;\nconst DATA_API_KEY$2 = '.data-api';\nconst ESCAPE_KEY$1 = 'Escape';\nconst EVENT_HIDE$4 = `hide${EVENT_KEY$4}`;\nconst EVENT_HIDE_PREVENTED$1 = `hidePrevented${EVENT_KEY$4}`;\nconst EVENT_HIDDEN$4 = `hidden${EVENT_KEY$4}`;\nconst EVENT_SHOW$4 = `show${EVENT_KEY$4}`;\nconst EVENT_SHOWN$4 = `shown${EVENT_KEY$4}`;\nconst EVENT_RESIZE$1 = `resize${EVENT_KEY$4}`;\nconst EVENT_CLICK_DISMISS = `click.dismiss${EVENT_KEY$4}`;\nconst EVENT_MOUSEDOWN_DISMISS = `mousedown.dismiss${EVENT_KEY$4}`;\nconst EVENT_KEYDOWN_DISMISS$1 = `keydown.dismiss${EVENT_KEY$4}`;\nconst EVENT_CLICK_DATA_API$2 = `click${EVENT_KEY$4}${DATA_API_KEY$2}`;\nconst CLASS_NAME_OPEN = 'modal-open';\nconst CLASS_NAME_FADE$3 = 'fade';\nconst CLASS_NAME_SHOW$4 = 'show';\nconst CLASS_NAME_STATIC = 'modal-static';\nconst OPEN_SELECTOR$1 = '.modal.show';\nconst SELECTOR_DIALOG = '.modal-dialog';\nconst SELECTOR_MODAL_BODY = '.modal-body';\nconst SELECTOR_DATA_TOGGLE$2 = '[data-bs-toggle=\"modal\"]';\nconst Default$6 = {\n backdrop: true,\n focus: true,\n keyboard: true\n};\nconst DefaultType$6 = {\n backdrop: '(boolean|string)',\n focus: 'boolean',\n keyboard: 'boolean'\n};\n\n/**\n * Class definition\n */\n\nclass Modal extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._dialog = SelectorEngine.findOne(SELECTOR_DIALOG, this._element);\n this._backdrop = this._initializeBackDrop();\n this._focustrap = this._initializeFocusTrap();\n this._isShown = false;\n this._isTransitioning = false;\n this._scrollBar = new ScrollBarHelper();\n this._addEventListeners();\n }\n\n // Getters\n static get Default() {\n return Default$6;\n }\n static get DefaultType() {\n return DefaultType$6;\n }\n static get NAME() {\n return NAME$7;\n }\n\n // Public\n toggle(relatedTarget) {\n return this._isShown ? this.hide() : this.show(relatedTarget);\n }\n show(relatedTarget) {\n if (this._isShown || this._isTransitioning) {\n return;\n }\n const showEvent = EventHandler.trigger(this._element, EVENT_SHOW$4, {\n relatedTarget\n });\n if (showEvent.defaultPrevented) {\n return;\n }\n this._isShown = true;\n this._isTransitioning = true;\n this._scrollBar.hide();\n document.body.classList.add(CLASS_NAME_OPEN);\n this._adjustDialog();\n this._backdrop.show(() => this._showElement(relatedTarget));\n }\n hide() {\n if (!this._isShown || this._isTransitioning) {\n return;\n }\n const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE$4);\n if (hideEvent.defaultPrevented) {\n return;\n }\n this._isShown = false;\n this._isTransitioning = true;\n this._focustrap.deactivate();\n this._element.classList.remove(CLASS_NAME_SHOW$4);\n this._queueCallback(() => this._hideModal(), this._element, this._isAnimated());\n }\n dispose() {\n EventHandler.off(window, EVENT_KEY$4);\n EventHandler.off(this._dialog, EVENT_KEY$4);\n this._backdrop.dispose();\n this._focustrap.deactivate();\n super.dispose();\n }\n handleUpdate() {\n this._adjustDialog();\n }\n\n // Private\n _initializeBackDrop() {\n return new Backdrop({\n isVisible: Boolean(this._config.backdrop),\n // 'static' option will be translated to true, and booleans will keep their value,\n isAnimated: this._isAnimated()\n });\n }\n _initializeFocusTrap() {\n return new FocusTrap({\n trapElement: this._element\n });\n }\n _showElement(relatedTarget) {\n // try to append dynamic modal\n if (!document.body.contains(this._element)) {\n document.body.append(this._element);\n }\n this._element.style.display = 'block';\n this._element.removeAttribute('aria-hidden');\n this._element.setAttribute('aria-modal', true);\n this._element.setAttribute('role', 'dialog');\n this._element.scrollTop = 0;\n const modalBody = SelectorEngine.findOne(SELECTOR_MODAL_BODY, this._dialog);\n if (modalBody) {\n modalBody.scrollTop = 0;\n }\n reflow(this._element);\n this._element.classList.add(CLASS_NAME_SHOW$4);\n const transitionComplete = () => {\n if (this._config.focus) {\n this._focustrap.activate();\n }\n this._isTransitioning = false;\n EventHandler.trigger(this._element, EVENT_SHOWN$4, {\n relatedTarget\n });\n };\n this._queueCallback(transitionComplete, this._dialog, this._isAnimated());\n }\n _addEventListeners() {\n EventHandler.on(this._element, EVENT_KEYDOWN_DISMISS$1, event => {\n if (event.key !== ESCAPE_KEY$1) {\n return;\n }\n if (this._config.keyboard) {\n this.hide();\n return;\n }\n this._triggerBackdropTransition();\n });\n EventHandler.on(window, EVENT_RESIZE$1, () => {\n if (this._isShown && !this._isTransitioning) {\n this._adjustDialog();\n }\n });\n EventHandler.on(this._element, EVENT_MOUSEDOWN_DISMISS, event => {\n // a bad trick to segregate clicks that may start inside dialog but end outside, and avoid listen to scrollbar clicks\n EventHandler.one(this._element, EVENT_CLICK_DISMISS, event2 => {\n if (this._element !== event.target || this._element !== event2.target) {\n return;\n }\n if (this._config.backdrop === 'static') {\n this._triggerBackdropTransition();\n return;\n }\n if (this._config.backdrop) {\n this.hide();\n }\n });\n });\n }\n _hideModal() {\n this._element.style.display = 'none';\n this._element.setAttribute('aria-hidden', true);\n this._element.removeAttribute('aria-modal');\n this._element.removeAttribute('role');\n this._isTransitioning = false;\n this._backdrop.hide(() => {\n document.body.classList.remove(CLASS_NAME_OPEN);\n this._resetAdjustments();\n this._scrollBar.reset();\n EventHandler.trigger(this._element, EVENT_HIDDEN$4);\n });\n }\n _isAnimated() {\n return this._element.classList.contains(CLASS_NAME_FADE$3);\n }\n _triggerBackdropTransition() {\n const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE_PREVENTED$1);\n if (hideEvent.defaultPrevented) {\n return;\n }\n const isModalOverflowing = this._element.scrollHeight > document.documentElement.clientHeight;\n const initialOverflowY = this._element.style.overflowY;\n // return if the following background transition hasn't yet completed\n if (initialOverflowY === 'hidden' || this._element.classList.contains(CLASS_NAME_STATIC)) {\n return;\n }\n if (!isModalOverflowing) {\n this._element.style.overflowY = 'hidden';\n }\n this._element.classList.add(CLASS_NAME_STATIC);\n this._queueCallback(() => {\n this._element.classList.remove(CLASS_NAME_STATIC);\n this._queueCallback(() => {\n this._element.style.overflowY = initialOverflowY;\n }, this._dialog);\n }, this._dialog);\n this._element.focus();\n }\n\n /**\n * The following methods are used to handle overflowing modals\n */\n\n _adjustDialog() {\n const isModalOverflowing = this._element.scrollHeight > document.documentElement.clientHeight;\n const scrollbarWidth = this._scrollBar.getWidth();\n const isBodyOverflowing = scrollbarWidth > 0;\n if (isBodyOverflowing && !isModalOverflowing) {\n const property = isRTL() ? 'paddingLeft' : 'paddingRight';\n this._element.style[property] = `${scrollbarWidth}px`;\n }\n if (!isBodyOverflowing && isModalOverflowing) {\n const property = isRTL() ? 'paddingRight' : 'paddingLeft';\n this._element.style[property] = `${scrollbarWidth}px`;\n }\n }\n _resetAdjustments() {\n this._element.style.paddingLeft = '';\n this._element.style.paddingRight = '';\n }\n\n // Static\n static jQueryInterface(config, relatedTarget) {\n return this.each(function () {\n const data = Modal.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config](relatedTarget);\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$2, SELECTOR_DATA_TOGGLE$2, function (event) {\n const target = SelectorEngine.getElementFromSelector(this);\n if (['A', 'AREA'].includes(this.tagName)) {\n event.preventDefault();\n }\n EventHandler.one(target, EVENT_SHOW$4, showEvent => {\n if (showEvent.defaultPrevented) {\n // only register focus restorer if modal will actually get shown\n return;\n }\n EventHandler.one(target, EVENT_HIDDEN$4, () => {\n if (isVisible(this)) {\n this.focus();\n }\n });\n });\n\n // avoid conflict when clicking modal toggler while another one is open\n const alreadyOpen = SelectorEngine.findOne(OPEN_SELECTOR$1);\n if (alreadyOpen) {\n Modal.getInstance(alreadyOpen).hide();\n }\n const data = Modal.getOrCreateInstance(target);\n data.toggle(this);\n});\nenableDismissTrigger(Modal);\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Modal);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap offcanvas.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$6 = 'offcanvas';\nconst DATA_KEY$3 = 'bs.offcanvas';\nconst EVENT_KEY$3 = `.${DATA_KEY$3}`;\nconst DATA_API_KEY$1 = '.data-api';\nconst EVENT_LOAD_DATA_API$2 = `load${EVENT_KEY$3}${DATA_API_KEY$1}`;\nconst ESCAPE_KEY = 'Escape';\nconst CLASS_NAME_SHOW$3 = 'show';\nconst CLASS_NAME_SHOWING$1 = 'showing';\nconst CLASS_NAME_HIDING = 'hiding';\nconst CLASS_NAME_BACKDROP = 'offcanvas-backdrop';\nconst OPEN_SELECTOR = '.offcanvas.show';\nconst EVENT_SHOW$3 = `show${EVENT_KEY$3}`;\nconst EVENT_SHOWN$3 = `shown${EVENT_KEY$3}`;\nconst EVENT_HIDE$3 = `hide${EVENT_KEY$3}`;\nconst EVENT_HIDE_PREVENTED = `hidePrevented${EVENT_KEY$3}`;\nconst EVENT_HIDDEN$3 = `hidden${EVENT_KEY$3}`;\nconst EVENT_RESIZE = `resize${EVENT_KEY$3}`;\nconst EVENT_CLICK_DATA_API$1 = `click${EVENT_KEY$3}${DATA_API_KEY$1}`;\nconst EVENT_KEYDOWN_DISMISS = `keydown.dismiss${EVENT_KEY$3}`;\nconst SELECTOR_DATA_TOGGLE$1 = '[data-bs-toggle=\"offcanvas\"]';\nconst Default$5 = {\n backdrop: true,\n keyboard: true,\n scroll: false\n};\nconst DefaultType$5 = {\n backdrop: '(boolean|string)',\n keyboard: 'boolean',\n scroll: 'boolean'\n};\n\n/**\n * Class definition\n */\n\nclass Offcanvas extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._isShown = false;\n this._backdrop = this._initializeBackDrop();\n this._focustrap = this._initializeFocusTrap();\n this._addEventListeners();\n }\n\n // Getters\n static get Default() {\n return Default$5;\n }\n static get DefaultType() {\n return DefaultType$5;\n }\n static get NAME() {\n return NAME$6;\n }\n\n // Public\n toggle(relatedTarget) {\n return this._isShown ? this.hide() : this.show(relatedTarget);\n }\n show(relatedTarget) {\n if (this._isShown) {\n return;\n }\n const showEvent = EventHandler.trigger(this._element, EVENT_SHOW$3, {\n relatedTarget\n });\n if (showEvent.defaultPrevented) {\n return;\n }\n this._isShown = true;\n this._backdrop.show();\n if (!this._config.scroll) {\n new ScrollBarHelper().hide();\n }\n this._element.setAttribute('aria-modal', true);\n this._element.setAttribute('role', 'dialog');\n this._element.classList.add(CLASS_NAME_SHOWING$1);\n const completeCallBack = () => {\n if (!this._config.scroll || this._config.backdrop) {\n this._focustrap.activate();\n }\n this._element.classList.add(CLASS_NAME_SHOW$3);\n this._element.classList.remove(CLASS_NAME_SHOWING$1);\n EventHandler.trigger(this._element, EVENT_SHOWN$3, {\n relatedTarget\n });\n };\n this._queueCallback(completeCallBack, this._element, true);\n }\n hide() {\n if (!this._isShown) {\n return;\n }\n const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE$3);\n if (hideEvent.defaultPrevented) {\n return;\n }\n this._focustrap.deactivate();\n this._element.blur();\n this._isShown = false;\n this._element.classList.add(CLASS_NAME_HIDING);\n this._backdrop.hide();\n const completeCallback = () => {\n this._element.classList.remove(CLASS_NAME_SHOW$3, CLASS_NAME_HIDING);\n this._element.removeAttribute('aria-modal');\n this._element.removeAttribute('role');\n if (!this._config.scroll) {\n new ScrollBarHelper().reset();\n }\n EventHandler.trigger(this._element, EVENT_HIDDEN$3);\n };\n this._queueCallback(completeCallback, this._element, true);\n }\n dispose() {\n this._backdrop.dispose();\n this._focustrap.deactivate();\n super.dispose();\n }\n\n // Private\n _initializeBackDrop() {\n const clickCallback = () => {\n if (this._config.backdrop === 'static') {\n EventHandler.trigger(this._element, EVENT_HIDE_PREVENTED);\n return;\n }\n this.hide();\n };\n\n // 'static' option will be translated to true, and booleans will keep their value\n const isVisible = Boolean(this._config.backdrop);\n return new Backdrop({\n className: CLASS_NAME_BACKDROP,\n isVisible,\n isAnimated: true,\n rootElement: this._element.parentNode,\n clickCallback: isVisible ? clickCallback : null\n });\n }\n _initializeFocusTrap() {\n return new FocusTrap({\n trapElement: this._element\n });\n }\n _addEventListeners() {\n EventHandler.on(this._element, EVENT_KEYDOWN_DISMISS, event => {\n if (event.key !== ESCAPE_KEY) {\n return;\n }\n if (this._config.keyboard) {\n this.hide();\n return;\n }\n EventHandler.trigger(this._element, EVENT_HIDE_PREVENTED);\n });\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Offcanvas.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (data[config] === undefined || config.startsWith('_') || config === 'constructor') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config](this);\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$1, SELECTOR_DATA_TOGGLE$1, function (event) {\n const target = SelectorEngine.getElementFromSelector(this);\n if (['A', 'AREA'].includes(this.tagName)) {\n event.preventDefault();\n }\n if (isDisabled(this)) {\n return;\n }\n EventHandler.one(target, EVENT_HIDDEN$3, () => {\n // focus on trigger when it is closed\n if (isVisible(this)) {\n this.focus();\n }\n });\n\n // avoid conflict when clicking a toggler of an offcanvas, while another is open\n const alreadyOpen = SelectorEngine.findOne(OPEN_SELECTOR);\n if (alreadyOpen && alreadyOpen !== target) {\n Offcanvas.getInstance(alreadyOpen).hide();\n }\n const data = Offcanvas.getOrCreateInstance(target);\n data.toggle(this);\n});\nEventHandler.on(window, EVENT_LOAD_DATA_API$2, () => {\n for (const selector of SelectorEngine.find(OPEN_SELECTOR)) {\n Offcanvas.getOrCreateInstance(selector).show();\n }\n});\nEventHandler.on(window, EVENT_RESIZE, () => {\n for (const element of SelectorEngine.find('[aria-modal][class*=show][class*=offcanvas-]')) {\n if (getComputedStyle(element).position !== 'fixed') {\n Offcanvas.getOrCreateInstance(element).hide();\n }\n }\n});\nenableDismissTrigger(Offcanvas);\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Offcanvas);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/sanitizer.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n// js-docs-start allow-list\nconst ARIA_ATTRIBUTE_PATTERN = /^aria-[\\w-]*$/i;\nconst DefaultAllowlist = {\n // Global attributes allowed on any supplied element below.\n '*': ['class', 'dir', 'id', 'lang', 'role', ARIA_ATTRIBUTE_PATTERN],\n a: ['target', 'href', 'title', 'rel'],\n area: [],\n b: [],\n br: [],\n col: [],\n code: [],\n dd: [],\n div: [],\n dl: [],\n dt: [],\n em: [],\n hr: [],\n h1: [],\n h2: [],\n h3: [],\n h4: [],\n h5: [],\n h6: [],\n i: [],\n img: ['src', 'srcset', 'alt', 'title', 'width', 'height'],\n li: [],\n ol: [],\n p: [],\n pre: [],\n s: [],\n small: [],\n span: [],\n sub: [],\n sup: [],\n strong: [],\n u: [],\n ul: []\n};\n// js-docs-end allow-list\n\nconst uriAttributes = new Set(['background', 'cite', 'href', 'itemtype', 'longdesc', 'poster', 'src', 'xlink:href']);\n\n/**\n * A pattern that recognizes URLs that are safe wrt. XSS in URL navigation\n * contexts.\n *\n * Shout-out to Angular https://github.com/angular/angular/blob/15.2.8/packages/core/src/sanitization/url_sanitizer.ts#L38\n */\n// eslint-disable-next-line unicorn/better-regex\nconst SAFE_URL_PATTERN = /^(?!javascript:)(?:[a-z0-9+.-]+:|[^&:/?#]*(?:[/?#]|$))/i;\nconst allowedAttribute = (attribute, allowedAttributeList) => {\n const attributeName = attribute.nodeName.toLowerCase();\n if (allowedAttributeList.includes(attributeName)) {\n if (uriAttributes.has(attributeName)) {\n return Boolean(SAFE_URL_PATTERN.test(attribute.nodeValue));\n }\n return true;\n }\n\n // Check if a regular expression validates the attribute.\n return allowedAttributeList.filter(attributeRegex => attributeRegex instanceof RegExp).some(regex => regex.test(attributeName));\n};\nfunction sanitizeHtml(unsafeHtml, allowList, sanitizeFunction) {\n if (!unsafeHtml.length) {\n return unsafeHtml;\n }\n if (sanitizeFunction && typeof sanitizeFunction === 'function') {\n return sanitizeFunction(unsafeHtml);\n }\n const domParser = new window.DOMParser();\n const createdDocument = domParser.parseFromString(unsafeHtml, 'text/html');\n const elements = [].concat(...createdDocument.body.querySelectorAll('*'));\n for (const element of elements) {\n const elementName = element.nodeName.toLowerCase();\n if (!Object.keys(allowList).includes(elementName)) {\n element.remove();\n continue;\n }\n const attributeList = [].concat(...element.attributes);\n const allowedAttributes = [].concat(allowList['*'] || [], allowList[elementName] || []);\n for (const attribute of attributeList) {\n if (!allowedAttribute(attribute, allowedAttributes)) {\n element.removeAttribute(attribute.nodeName);\n }\n }\n }\n return createdDocument.body.innerHTML;\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/template-factory.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$5 = 'TemplateFactory';\nconst Default$4 = {\n allowList: DefaultAllowlist,\n content: {},\n // { selector : text , selector2 : text2 , }\n extraClass: '',\n html: false,\n sanitize: true,\n sanitizeFn: null,\n template: '
'\n};\nconst DefaultType$4 = {\n allowList: 'object',\n content: 'object',\n extraClass: '(string|function)',\n html: 'boolean',\n sanitize: 'boolean',\n sanitizeFn: '(null|function)',\n template: 'string'\n};\nconst DefaultContentType = {\n entry: '(string|element|function|null)',\n selector: '(string|element)'\n};\n\n/**\n * Class definition\n */\n\nclass TemplateFactory extends Config {\n constructor(config) {\n super();\n this._config = this._getConfig(config);\n }\n\n // Getters\n static get Default() {\n return Default$4;\n }\n static get DefaultType() {\n return DefaultType$4;\n }\n static get NAME() {\n return NAME$5;\n }\n\n // Public\n getContent() {\n return Object.values(this._config.content).map(config => this._resolvePossibleFunction(config)).filter(Boolean);\n }\n hasContent() {\n return this.getContent().length > 0;\n }\n changeContent(content) {\n this._checkContent(content);\n this._config.content = {\n ...this._config.content,\n ...content\n };\n return this;\n }\n toHtml() {\n const templateWrapper = document.createElement('div');\n templateWrapper.innerHTML = this._maybeSanitize(this._config.template);\n for (const [selector, text] of Object.entries(this._config.content)) {\n this._setContent(templateWrapper, text, selector);\n }\n const template = templateWrapper.children[0];\n const extraClass = this._resolvePossibleFunction(this._config.extraClass);\n if (extraClass) {\n template.classList.add(...extraClass.split(' '));\n }\n return template;\n }\n\n // Private\n _typeCheckConfig(config) {\n super._typeCheckConfig(config);\n this._checkContent(config.content);\n }\n _checkContent(arg) {\n for (const [selector, content] of Object.entries(arg)) {\n super._typeCheckConfig({\n selector,\n entry: content\n }, DefaultContentType);\n }\n }\n _setContent(template, content, selector) {\n const templateElement = SelectorEngine.findOne(selector, template);\n if (!templateElement) {\n return;\n }\n content = this._resolvePossibleFunction(content);\n if (!content) {\n templateElement.remove();\n return;\n }\n if (isElement(content)) {\n this._putElementInTemplate(getElement(content), templateElement);\n return;\n }\n if (this._config.html) {\n templateElement.innerHTML = this._maybeSanitize(content);\n return;\n }\n templateElement.textContent = content;\n }\n _maybeSanitize(arg) {\n return this._config.sanitize ? sanitizeHtml(arg, this._config.allowList, this._config.sanitizeFn) : arg;\n }\n _resolvePossibleFunction(arg) {\n return execute(arg, [this]);\n }\n _putElementInTemplate(element, templateElement) {\n if (this._config.html) {\n templateElement.innerHTML = '';\n templateElement.append(element);\n return;\n }\n templateElement.textContent = element.textContent;\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap tooltip.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$4 = 'tooltip';\nconst DISALLOWED_ATTRIBUTES = new Set(['sanitize', 'allowList', 'sanitizeFn']);\nconst CLASS_NAME_FADE$2 = 'fade';\nconst CLASS_NAME_MODAL = 'modal';\nconst CLASS_NAME_SHOW$2 = 'show';\nconst SELECTOR_TOOLTIP_INNER = '.tooltip-inner';\nconst SELECTOR_MODAL = `.${CLASS_NAME_MODAL}`;\nconst EVENT_MODAL_HIDE = 'hide.bs.modal';\nconst TRIGGER_HOVER = 'hover';\nconst TRIGGER_FOCUS = 'focus';\nconst TRIGGER_CLICK = 'click';\nconst TRIGGER_MANUAL = 'manual';\nconst EVENT_HIDE$2 = 'hide';\nconst EVENT_HIDDEN$2 = 'hidden';\nconst EVENT_SHOW$2 = 'show';\nconst EVENT_SHOWN$2 = 'shown';\nconst EVENT_INSERTED = 'inserted';\nconst EVENT_CLICK$1 = 'click';\nconst EVENT_FOCUSIN$1 = 'focusin';\nconst EVENT_FOCUSOUT$1 = 'focusout';\nconst EVENT_MOUSEENTER = 'mouseenter';\nconst EVENT_MOUSELEAVE = 'mouseleave';\nconst AttachmentMap = {\n AUTO: 'auto',\n TOP: 'top',\n RIGHT: isRTL() ? 'left' : 'right',\n BOTTOM: 'bottom',\n LEFT: isRTL() ? 'right' : 'left'\n};\nconst Default$3 = {\n allowList: DefaultAllowlist,\n animation: true,\n boundary: 'clippingParents',\n container: false,\n customClass: '',\n delay: 0,\n fallbackPlacements: ['top', 'right', 'bottom', 'left'],\n html: false,\n offset: [0, 6],\n placement: 'top',\n popperConfig: null,\n sanitize: true,\n sanitizeFn: null,\n selector: false,\n template: '
' + '
' + '
' + '
',\n title: '',\n trigger: 'hover focus'\n};\nconst DefaultType$3 = {\n allowList: 'object',\n animation: 'boolean',\n boundary: '(string|element)',\n container: '(string|element|boolean)',\n customClass: '(string|function)',\n delay: '(number|object)',\n fallbackPlacements: 'array',\n html: 'boolean',\n offset: '(array|string|function)',\n placement: '(string|function)',\n popperConfig: '(null|object|function)',\n sanitize: 'boolean',\n sanitizeFn: '(null|function)',\n selector: '(string|boolean)',\n template: 'string',\n title: '(string|element|function)',\n trigger: 'string'\n};\n\n/**\n * Class definition\n */\n\nclass Tooltip extends BaseComponent {\n constructor(element, config) {\n if (typeof Popper === 'undefined') {\n throw new TypeError('Bootstrap\\'s tooltips require Popper (https://popper.js.org)');\n }\n super(element, config);\n\n // Private\n this._isEnabled = true;\n this._timeout = 0;\n this._isHovered = null;\n this._activeTrigger = {};\n this._popper = null;\n this._templateFactory = null;\n this._newContent = null;\n\n // Protected\n this.tip = null;\n this._setListeners();\n if (!this._config.selector) {\n this._fixTitle();\n }\n }\n\n // Getters\n static get Default() {\n return Default$3;\n }\n static get DefaultType() {\n return DefaultType$3;\n }\n static get NAME() {\n return NAME$4;\n }\n\n // Public\n enable() {\n this._isEnabled = true;\n }\n disable() {\n this._isEnabled = false;\n }\n toggleEnabled() {\n this._isEnabled = !this._isEnabled;\n }\n toggle() {\n if (!this._isEnabled) {\n return;\n }\n this._activeTrigger.click = !this._activeTrigger.click;\n if (this._isShown()) {\n this._leave();\n return;\n }\n this._enter();\n }\n dispose() {\n clearTimeout(this._timeout);\n EventHandler.off(this._element.closest(SELECTOR_MODAL), EVENT_MODAL_HIDE, this._hideModalHandler);\n if (this._element.getAttribute('data-bs-original-title')) {\n this._element.setAttribute('title', this._element.getAttribute('data-bs-original-title'));\n }\n this._disposePopper();\n super.dispose();\n }\n show() {\n if (this._element.style.display === 'none') {\n throw new Error('Please use show on visible elements');\n }\n if (!(this._isWithContent() && this._isEnabled)) {\n return;\n }\n const showEvent = EventHandler.trigger(this._element, this.constructor.eventName(EVENT_SHOW$2));\n const shadowRoot = findShadowRoot(this._element);\n const isInTheDom = (shadowRoot || this._element.ownerDocument.documentElement).contains(this._element);\n if (showEvent.defaultPrevented || !isInTheDom) {\n return;\n }\n\n // TODO: v6 remove this or make it optional\n this._disposePopper();\n const tip = this._getTipElement();\n this._element.setAttribute('aria-describedby', tip.getAttribute('id'));\n const {\n container\n } = this._config;\n if (!this._element.ownerDocument.documentElement.contains(this.tip)) {\n container.append(tip);\n EventHandler.trigger(this._element, this.constructor.eventName(EVENT_INSERTED));\n }\n this._popper = this._createPopper(tip);\n tip.classList.add(CLASS_NAME_SHOW$2);\n\n // If this is a touch-enabled device we add extra\n // empty mouseover listeners to the body's immediate children;\n // only needed because of broken event delegation on iOS\n // https://www.quirksmode.org/blog/archives/2014/02/mouse_event_bub.html\n if ('ontouchstart' in document.documentElement) {\n for (const element of [].concat(...document.body.children)) {\n EventHandler.on(element, 'mouseover', noop);\n }\n }\n const complete = () => {\n EventHandler.trigger(this._element, this.constructor.eventName(EVENT_SHOWN$2));\n if (this._isHovered === false) {\n this._leave();\n }\n this._isHovered = false;\n };\n this._queueCallback(complete, this.tip, this._isAnimated());\n }\n hide() {\n if (!this._isShown()) {\n return;\n }\n const hideEvent = EventHandler.trigger(this._element, this.constructor.eventName(EVENT_HIDE$2));\n if (hideEvent.defaultPrevented) {\n return;\n }\n const tip = this._getTipElement();\n tip.classList.remove(CLASS_NAME_SHOW$2);\n\n // If this is a touch-enabled device we remove the extra\n // empty mouseover listeners we added for iOS support\n if ('ontouchstart' in document.documentElement) {\n for (const element of [].concat(...document.body.children)) {\n EventHandler.off(element, 'mouseover', noop);\n }\n }\n this._activeTrigger[TRIGGER_CLICK] = false;\n this._activeTrigger[TRIGGER_FOCUS] = false;\n this._activeTrigger[TRIGGER_HOVER] = false;\n this._isHovered = null; // it is a trick to support manual triggering\n\n const complete = () => {\n if (this._isWithActiveTrigger()) {\n return;\n }\n if (!this._isHovered) {\n this._disposePopper();\n }\n this._element.removeAttribute('aria-describedby');\n EventHandler.trigger(this._element, this.constructor.eventName(EVENT_HIDDEN$2));\n };\n this._queueCallback(complete, this.tip, this._isAnimated());\n }\n update() {\n if (this._popper) {\n this._popper.update();\n }\n }\n\n // Protected\n _isWithContent() {\n return Boolean(this._getTitle());\n }\n _getTipElement() {\n if (!this.tip) {\n this.tip = this._createTipElement(this._newContent || this._getContentForTemplate());\n }\n return this.tip;\n }\n _createTipElement(content) {\n const tip = this._getTemplateFactory(content).toHtml();\n\n // TODO: remove this check in v6\n if (!tip) {\n return null;\n }\n tip.classList.remove(CLASS_NAME_FADE$2, CLASS_NAME_SHOW$2);\n // TODO: v6 the following can be achieved with CSS only\n tip.classList.add(`bs-${this.constructor.NAME}-auto`);\n const tipId = getUID(this.constructor.NAME).toString();\n tip.setAttribute('id', tipId);\n if (this._isAnimated()) {\n tip.classList.add(CLASS_NAME_FADE$2);\n }\n return tip;\n }\n setContent(content) {\n this._newContent = content;\n if (this._isShown()) {\n this._disposePopper();\n this.show();\n }\n }\n _getTemplateFactory(content) {\n if (this._templateFactory) {\n this._templateFactory.changeContent(content);\n } else {\n this._templateFactory = new TemplateFactory({\n ...this._config,\n // the `content` var has to be after `this._config`\n // to override config.content in case of popover\n content,\n extraClass: this._resolvePossibleFunction(this._config.customClass)\n });\n }\n return this._templateFactory;\n }\n _getContentForTemplate() {\n return {\n [SELECTOR_TOOLTIP_INNER]: this._getTitle()\n };\n }\n _getTitle() {\n return this._resolvePossibleFunction(this._config.title) || this._element.getAttribute('data-bs-original-title');\n }\n\n // Private\n _initializeOnDelegatedTarget(event) {\n return this.constructor.getOrCreateInstance(event.delegateTarget, this._getDelegateConfig());\n }\n _isAnimated() {\n return this._config.animation || this.tip && this.tip.classList.contains(CLASS_NAME_FADE$2);\n }\n _isShown() {\n return this.tip && this.tip.classList.contains(CLASS_NAME_SHOW$2);\n }\n _createPopper(tip) {\n const placement = execute(this._config.placement, [this, tip, this._element]);\n const attachment = AttachmentMap[placement.toUpperCase()];\n return Popper.createPopper(this._element, tip, this._getPopperConfig(attachment));\n }\n _getOffset() {\n const {\n offset\n } = this._config;\n if (typeof offset === 'string') {\n return offset.split(',').map(value => Number.parseInt(value, 10));\n }\n if (typeof offset === 'function') {\n return popperData => offset(popperData, this._element);\n }\n return offset;\n }\n _resolvePossibleFunction(arg) {\n return execute(arg, [this._element]);\n }\n _getPopperConfig(attachment) {\n const defaultBsPopperConfig = {\n placement: attachment,\n modifiers: [{\n name: 'flip',\n options: {\n fallbackPlacements: this._config.fallbackPlacements\n }\n }, {\n name: 'offset',\n options: {\n offset: this._getOffset()\n }\n }, {\n name: 'preventOverflow',\n options: {\n boundary: this._config.boundary\n }\n }, {\n name: 'arrow',\n options: {\n element: `.${this.constructor.NAME}-arrow`\n }\n }, {\n name: 'preSetPlacement',\n enabled: true,\n phase: 'beforeMain',\n fn: data => {\n // Pre-set Popper's placement attribute in order to read the arrow sizes properly.\n // Otherwise, Popper mixes up the width and height dimensions since the initial arrow style is for top placement\n this._getTipElement().setAttribute('data-popper-placement', data.state.placement);\n }\n }]\n };\n return {\n ...defaultBsPopperConfig,\n ...execute(this._config.popperConfig, [defaultBsPopperConfig])\n };\n }\n _setListeners() {\n const triggers = this._config.trigger.split(' ');\n for (const trigger of triggers) {\n if (trigger === 'click') {\n EventHandler.on(this._element, this.constructor.eventName(EVENT_CLICK$1), this._config.selector, event => {\n const context = this._initializeOnDelegatedTarget(event);\n context.toggle();\n });\n } else if (trigger !== TRIGGER_MANUAL) {\n const eventIn = trigger === TRIGGER_HOVER ? this.constructor.eventName(EVENT_MOUSEENTER) : this.constructor.eventName(EVENT_FOCUSIN$1);\n const eventOut = trigger === TRIGGER_HOVER ? this.constructor.eventName(EVENT_MOUSELEAVE) : this.constructor.eventName(EVENT_FOCUSOUT$1);\n EventHandler.on(this._element, eventIn, this._config.selector, event => {\n const context = this._initializeOnDelegatedTarget(event);\n context._activeTrigger[event.type === 'focusin' ? TRIGGER_FOCUS : TRIGGER_HOVER] = true;\n context._enter();\n });\n EventHandler.on(this._element, eventOut, this._config.selector, event => {\n const context = this._initializeOnDelegatedTarget(event);\n context._activeTrigger[event.type === 'focusout' ? TRIGGER_FOCUS : TRIGGER_HOVER] = context._element.contains(event.relatedTarget);\n context._leave();\n });\n }\n }\n this._hideModalHandler = () => {\n if (this._element) {\n this.hide();\n }\n };\n EventHandler.on(this._element.closest(SELECTOR_MODAL), EVENT_MODAL_HIDE, this._hideModalHandler);\n }\n _fixTitle() {\n const title = this._element.getAttribute('title');\n if (!title) {\n return;\n }\n if (!this._element.getAttribute('aria-label') && !this._element.textContent.trim()) {\n this._element.setAttribute('aria-label', title);\n }\n this._element.setAttribute('data-bs-original-title', title); // DO NOT USE IT. Is only for backwards compatibility\n this._element.removeAttribute('title');\n }\n _enter() {\n if (this._isShown() || this._isHovered) {\n this._isHovered = true;\n return;\n }\n this._isHovered = true;\n this._setTimeout(() => {\n if (this._isHovered) {\n this.show();\n }\n }, this._config.delay.show);\n }\n _leave() {\n if (this._isWithActiveTrigger()) {\n return;\n }\n this._isHovered = false;\n this._setTimeout(() => {\n if (!this._isHovered) {\n this.hide();\n }\n }, this._config.delay.hide);\n }\n _setTimeout(handler, timeout) {\n clearTimeout(this._timeout);\n this._timeout = setTimeout(handler, timeout);\n }\n _isWithActiveTrigger() {\n return Object.values(this._activeTrigger).includes(true);\n }\n _getConfig(config) {\n const dataAttributes = Manipulator.getDataAttributes(this._element);\n for (const dataAttribute of Object.keys(dataAttributes)) {\n if (DISALLOWED_ATTRIBUTES.has(dataAttribute)) {\n delete dataAttributes[dataAttribute];\n }\n }\n config = {\n ...dataAttributes,\n ...(typeof config === 'object' && config ? config : {})\n };\n config = this._mergeConfigObj(config);\n config = this._configAfterMerge(config);\n this._typeCheckConfig(config);\n return config;\n }\n _configAfterMerge(config) {\n config.container = config.container === false ? document.body : getElement(config.container);\n if (typeof config.delay === 'number') {\n config.delay = {\n show: config.delay,\n hide: config.delay\n };\n }\n if (typeof config.title === 'number') {\n config.title = config.title.toString();\n }\n if (typeof config.content === 'number') {\n config.content = config.content.toString();\n }\n return config;\n }\n _getDelegateConfig() {\n const config = {};\n for (const [key, value] of Object.entries(this._config)) {\n if (this.constructor.Default[key] !== value) {\n config[key] = value;\n }\n }\n config.selector = false;\n config.trigger = 'manual';\n\n // In the future can be replaced with:\n // const keysWithDifferentValues = Object.entries(this._config).filter(entry => this.constructor.Default[entry[0]] !== this._config[entry[0]])\n // `Object.fromEntries(keysWithDifferentValues)`\n return config;\n }\n _disposePopper() {\n if (this._popper) {\n this._popper.destroy();\n this._popper = null;\n }\n if (this.tip) {\n this.tip.remove();\n this.tip = null;\n }\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Tooltip.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n });\n }\n}\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Tooltip);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap popover.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$3 = 'popover';\nconst SELECTOR_TITLE = '.popover-header';\nconst SELECTOR_CONTENT = '.popover-body';\nconst Default$2 = {\n ...Tooltip.Default,\n content: '',\n offset: [0, 8],\n placement: 'right',\n template: '
' + '
' + '

' + '
' + '
',\n trigger: 'click'\n};\nconst DefaultType$2 = {\n ...Tooltip.DefaultType,\n content: '(null|string|element|function)'\n};\n\n/**\n * Class definition\n */\n\nclass Popover extends Tooltip {\n // Getters\n static get Default() {\n return Default$2;\n }\n static get DefaultType() {\n return DefaultType$2;\n }\n static get NAME() {\n return NAME$3;\n }\n\n // Overrides\n _isWithContent() {\n return this._getTitle() || this._getContent();\n }\n\n // Private\n _getContentForTemplate() {\n return {\n [SELECTOR_TITLE]: this._getTitle(),\n [SELECTOR_CONTENT]: this._getContent()\n };\n }\n _getContent() {\n return this._resolvePossibleFunction(this._config.content);\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Popover.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n });\n }\n}\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Popover);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap scrollspy.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$2 = 'scrollspy';\nconst DATA_KEY$2 = 'bs.scrollspy';\nconst EVENT_KEY$2 = `.${DATA_KEY$2}`;\nconst DATA_API_KEY = '.data-api';\nconst EVENT_ACTIVATE = `activate${EVENT_KEY$2}`;\nconst EVENT_CLICK = `click${EVENT_KEY$2}`;\nconst EVENT_LOAD_DATA_API$1 = `load${EVENT_KEY$2}${DATA_API_KEY}`;\nconst CLASS_NAME_DROPDOWN_ITEM = 'dropdown-item';\nconst CLASS_NAME_ACTIVE$1 = 'active';\nconst SELECTOR_DATA_SPY = '[data-bs-spy=\"scroll\"]';\nconst SELECTOR_TARGET_LINKS = '[href]';\nconst SELECTOR_NAV_LIST_GROUP = '.nav, .list-group';\nconst SELECTOR_NAV_LINKS = '.nav-link';\nconst SELECTOR_NAV_ITEMS = '.nav-item';\nconst SELECTOR_LIST_ITEMS = '.list-group-item';\nconst SELECTOR_LINK_ITEMS = `${SELECTOR_NAV_LINKS}, ${SELECTOR_NAV_ITEMS} > ${SELECTOR_NAV_LINKS}, ${SELECTOR_LIST_ITEMS}`;\nconst SELECTOR_DROPDOWN = '.dropdown';\nconst SELECTOR_DROPDOWN_TOGGLE$1 = '.dropdown-toggle';\nconst Default$1 = {\n offset: null,\n // TODO: v6 @deprecated, keep it for backwards compatibility reasons\n rootMargin: '0px 0px -25%',\n smoothScroll: false,\n target: null,\n threshold: [0.1, 0.5, 1]\n};\nconst DefaultType$1 = {\n offset: '(number|null)',\n // TODO v6 @deprecated, keep it for backwards compatibility reasons\n rootMargin: 'string',\n smoothScroll: 'boolean',\n target: 'element',\n threshold: 'array'\n};\n\n/**\n * Class definition\n */\n\nclass ScrollSpy extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n\n // this._element is the observablesContainer and config.target the menu links wrapper\n this._targetLinks = new Map();\n this._observableSections = new Map();\n this._rootElement = getComputedStyle(this._element).overflowY === 'visible' ? null : this._element;\n this._activeTarget = null;\n this._observer = null;\n this._previousScrollData = {\n visibleEntryTop: 0,\n parentScrollTop: 0\n };\n this.refresh(); // initialize\n }\n\n // Getters\n static get Default() {\n return Default$1;\n }\n static get DefaultType() {\n return DefaultType$1;\n }\n static get NAME() {\n return NAME$2;\n }\n\n // Public\n refresh() {\n this._initializeTargetsAndObservables();\n this._maybeEnableSmoothScroll();\n if (this._observer) {\n this._observer.disconnect();\n } else {\n this._observer = this._getNewObserver();\n }\n for (const section of this._observableSections.values()) {\n this._observer.observe(section);\n }\n }\n dispose() {\n this._observer.disconnect();\n super.dispose();\n }\n\n // Private\n _configAfterMerge(config) {\n // TODO: on v6 target should be given explicitly & remove the {target: 'ss-target'} case\n config.target = getElement(config.target) || document.body;\n\n // TODO: v6 Only for backwards compatibility reasons. Use rootMargin only\n config.rootMargin = config.offset ? `${config.offset}px 0px -30%` : config.rootMargin;\n if (typeof config.threshold === 'string') {\n config.threshold = config.threshold.split(',').map(value => Number.parseFloat(value));\n }\n return config;\n }\n _maybeEnableSmoothScroll() {\n if (!this._config.smoothScroll) {\n return;\n }\n\n // unregister any previous listeners\n EventHandler.off(this._config.target, EVENT_CLICK);\n EventHandler.on(this._config.target, EVENT_CLICK, SELECTOR_TARGET_LINKS, event => {\n const observableSection = this._observableSections.get(event.target.hash);\n if (observableSection) {\n event.preventDefault();\n const root = this._rootElement || window;\n const height = observableSection.offsetTop - this._element.offsetTop;\n if (root.scrollTo) {\n root.scrollTo({\n top: height,\n behavior: 'smooth'\n });\n return;\n }\n\n // Chrome 60 doesn't support `scrollTo`\n root.scrollTop = height;\n }\n });\n }\n _getNewObserver() {\n const options = {\n root: this._rootElement,\n threshold: this._config.threshold,\n rootMargin: this._config.rootMargin\n };\n return new IntersectionObserver(entries => this._observerCallback(entries), options);\n }\n\n // The logic of selection\n _observerCallback(entries) {\n const targetElement = entry => this._targetLinks.get(`#${entry.target.id}`);\n const activate = entry => {\n this._previousScrollData.visibleEntryTop = entry.target.offsetTop;\n this._process(targetElement(entry));\n };\n const parentScrollTop = (this._rootElement || document.documentElement).scrollTop;\n const userScrollsDown = parentScrollTop >= this._previousScrollData.parentScrollTop;\n this._previousScrollData.parentScrollTop = parentScrollTop;\n for (const entry of entries) {\n if (!entry.isIntersecting) {\n this._activeTarget = null;\n this._clearActiveClass(targetElement(entry));\n continue;\n }\n const entryIsLowerThanPrevious = entry.target.offsetTop >= this._previousScrollData.visibleEntryTop;\n // if we are scrolling down, pick the bigger offsetTop\n if (userScrollsDown && entryIsLowerThanPrevious) {\n activate(entry);\n // if parent isn't scrolled, let's keep the first visible item, breaking the iteration\n if (!parentScrollTop) {\n return;\n }\n continue;\n }\n\n // if we are scrolling up, pick the smallest offsetTop\n if (!userScrollsDown && !entryIsLowerThanPrevious) {\n activate(entry);\n }\n }\n }\n _initializeTargetsAndObservables() {\n this._targetLinks = new Map();\n this._observableSections = new Map();\n const targetLinks = SelectorEngine.find(SELECTOR_TARGET_LINKS, this._config.target);\n for (const anchor of targetLinks) {\n // ensure that the anchor has an id and is not disabled\n if (!anchor.hash || isDisabled(anchor)) {\n continue;\n }\n const observableSection = SelectorEngine.findOne(decodeURI(anchor.hash), this._element);\n\n // ensure that the observableSection exists & is visible\n if (isVisible(observableSection)) {\n this._targetLinks.set(decodeURI(anchor.hash), anchor);\n this._observableSections.set(anchor.hash, observableSection);\n }\n }\n }\n _process(target) {\n if (this._activeTarget === target) {\n return;\n }\n this._clearActiveClass(this._config.target);\n this._activeTarget = target;\n target.classList.add(CLASS_NAME_ACTIVE$1);\n this._activateParents(target);\n EventHandler.trigger(this._element, EVENT_ACTIVATE, {\n relatedTarget: target\n });\n }\n _activateParents(target) {\n // Activate dropdown parents\n if (target.classList.contains(CLASS_NAME_DROPDOWN_ITEM)) {\n SelectorEngine.findOne(SELECTOR_DROPDOWN_TOGGLE$1, target.closest(SELECTOR_DROPDOWN)).classList.add(CLASS_NAME_ACTIVE$1);\n return;\n }\n for (const listGroup of SelectorEngine.parents(target, SELECTOR_NAV_LIST_GROUP)) {\n // Set triggered links parents as active\n // With both