frontend/src/routes/potus-tracker/+page.svelte
MuslemRahimi 9c4429cbc0 ui fix
2025-03-28 11:36:45 +01:00

1072 lines
42 KiB
Svelte

<script lang="ts">
import SEO from "$lib/components/SEO.svelte";
import Infobox from "$lib/components/Infobox.svelte";
import highcharts from "$lib/highcharts.ts";
import * as DropdownMenu from "$lib/components/shadcn/dropdown-menu/index.js";
import { Button } from "$lib/components/shadcn/button/index.js";
import { sectorList } from "$lib/utils";
import avatar from "$lib/images/trump-avatar.jpeg";
import { mode } from "mode-watcher";
import { goto } from "$app/navigation";
import html2canvas from "html2canvas-pro";
export let data;
let postContent = "n/a";
let postDate = "n/a";
let postUrl = "#";
let postTitle = "n/a";
const updatedSectorList = ["S&P500", ...sectorList];
let rawData = data?.getData?.history || [];
let posts = data?.getData?.posts || [];
let executiveOrders = data?.getData?.executiveOrders || [];
let selectedSector = "S&P500";
const sectorDict = {
"S&P500": "SPY",
"Basic Materials": "XLB",
"Communication Services": "XLC",
Energy: "XLE",
"Financial Services": "XLF",
Industrials: "XLI",
Technology: "XLK",
"Consumer Defensive": "XLP",
"Real Estate": "XLRE",
Utilities: "XLU",
Healthcare: "XLV",
"Consumer Cyclical": "XLY",
};
const groupedByDate = rawData?.reduce((acc, item) => {
const dateKey = new Intl.DateTimeFormat("en-US", {
day: "2-digit",
month: "short",
year: "numeric",
timeZone: "UTC",
}).format(new Date(item?.date));
if (!acc[dateKey]) acc[dateKey] = [];
acc[dateKey].push(item);
return acc;
}, {});
let groupedOrders = executiveOrders?.reduce((acc, item) => {
const dateKey = new Intl.DateTimeFormat("en-US", {
day: "2-digit",
month: "short",
year: "numeric",
timeZone: "UTC",
}).format(new Date(item.date));
if (!acc[dateKey]) acc[dateKey] = [];
acc[dateKey].push(item);
return acc;
}, {});
const tabs = [
{
title: "Presidential Schedule",
},
{
title: "Executive Orders",
},
{
title: "Truth Social Post",
},
];
let activeIdx = 0;
let expandedDescriptions: { [key: string]: boolean } = {};
function truncateText(text: string, maxLength: number = 300) {
if (text.length <= maxLength) return text;
return text.slice(0, maxLength) + "...";
}
function latestInfoDate(inputDate) {
// Create a Date object for the input date and convert it to New York time zone
const inputDateLocal = new Date(inputDate).toLocaleString("en-US", {
timeZone: "America/New_York",
});
// Get the current date and time in New York timezone
const todayLocal = new Date().toLocaleString("en-US", {
timeZone: "America/New_York",
});
// Convert the localized strings back to Date objects
const inputDateMs = new Date(inputDateLocal).getTime();
const todayMs = new Date(todayLocal).getTime();
// Calculate the difference in milliseconds
const differenceInMs = todayMs - inputDateMs;
// Convert milliseconds to days
const differenceInDays = Math.floor(differenceInMs / (1000 * 60 * 60 * 24));
// Return whether the difference is less than or equal to 1 day
return differenceInDays <= 2;
}
function plotData() {
// Transform raw data into arrays for Highcharts
const rawData =
data?.getData?.marketPerformance[sectorDict[selectedSector]];
// Convert data into arrays for Highcharts
const categories = Object.keys(rawData);
const values = Object.values(rawData);
const options = {
credits: {
enabled: false,
},
chart: {
type: "column",
backgroundColor: $mode === "light" ? "#fff" : "#09090B",
plotBackgroundColor: $mode === "light" ? "#fff" : "#09090B",
height: 360,
animation: false,
},
title: {
text: `<h3 class="mt-3 mb-1 text-[1rem] sm:text-xl">${selectedSector} - Performance</h3>`,
style: {
color: $mode === "light" ? "black" : "white",
},
useHTML: true,
},
xAxis: {
categories: categories,
gridLineWidth: 0,
labels: {
style: { color: $mode === "light" ? "black" : "white" },
},
},
yAxis: {
gridLineWidth: 1,
gridLineColor: $mode === "light" ? "#d1d5dc" : "#111827",
labels: {
style: { color: $mode === "light" ? "black" : "white" },
formatter: function () {
return this.value + "%"; // Add percentage symbol
},
},
title: { text: null },
opposite: true,
},
tooltip: {
enabled: false,
},
plotOptions: {
series: {
color: $mode === "light" ? "black" : "white",
animation: false,
dataLabels: {
enabled: true,
color: $mode === "light" ? "black" : "white",
style: {
fontSize: "14px",
fontWeight: "normal",
},
formatter: function () {
return this.y.toFixed(2) + "%"; // Add percentage symbol
},
},
},
},
legend: {
enabled: false,
},
series: [
{
name: "Performance",
data: values,
zones: [
{
value: 0, // Values below 0
color: "#E02424", // Red
borderRadius: "0px",
borderColor: "#E02424", // Red border
},
{
color: "#10B981", // Green for values 0 and above
borderColor: "#10B981", // Green border
borderRadius: "0px",
},
],
},
],
};
return options;
}
let config = null;
async function captureScreenshot(index) {
const postElement = document.querySelector(`#post-${index}`);
// Clone the element to avoid modifying the original
const clonedElement = postElement.cloneNode(true);
// Create a temporary container for the clone with fixed dimensions and styling
const tempContainer = document.createElement("div");
tempContainer.style.position = "absolute";
tempContainer.style.left = "-9999px";
tempContainer.style.top = "0";
tempContainer.appendChild(clonedElement);
document.body.appendChild(tempContainer);
// Force light mode styling explicitly
clonedElement.style.cssText =
"background-color: white !important; color: black !important;";
// Process all elements recursively to ensure text visibility
function forceVisibleText(element) {
// Apply styles directly
element.style.cssText +=
"; color: black !important; background-color: white !important; border-color: #ccc !important;";
// For SVG elements, ensure they're visible
if (
element.tagName.toLowerCase() === "svg" ||
element.tagName.toLowerCase() === "path"
) {
element.style.cssText +=
"; fill: #333 !important; stroke: #333 !important;";
}
// Remove problematic classes completely instead of just removing dark: prefix
if (element.classList) {
const classesToRemove = [];
element.classList.forEach((cls) => {
if (
cls.includes("dark:") ||
cls.includes("text-gray") ||
cls.includes("text-white")
) {
classesToRemove.push(cls);
}
});
classesToRemove.forEach((cls) => element.classList.remove(cls));
// Add explicit light mode classes
element.classList.add("text-black");
}
// Process all child elements
if (element.children && element.children.length > 0) {
Array.from(element.children).forEach((child) =>
forceVisibleText(child),
);
}
}
// Apply the text visibility fix to all elements
forceVisibleText(clonedElement);
// Additional specific fixes for elements that might still have issues
const allTextElements = clonedElement.querySelectorAll(
"p, h1, h2, h3, h4, h5, span, a, div, label",
);
allTextElements.forEach((el) => {
el.style.color = "black";
el.setAttribute(
"style",
el.getAttribute("style") + "; color: black !important;",
);
});
// Convert any CSS variables that might affect color
const computed = window.getComputedStyle(postElement);
const cssText =
":root { --text-color: black !important; --background-color: white !important; }";
const style = document.createElement("style");
style.textContent = cssText;
clonedElement.appendChild(style);
// Wait a bit to ensure styles are applied
await new Promise((resolve) => setTimeout(resolve, 50));
// Capture screenshot
try {
const canvas = await html2canvas(clonedElement, {
backgroundColor: "white",
logging: true, // Enable logging to help debug
scale: 2, // Higher quality
useCORS: true,
allowTaint: true,
removeContainer: false, // Handle cleanup ourselves
});
// Convert to image
const image = canvas.toDataURL("image/png");
// Create a download link
const link = document.createElement("a");
link.href = image;
link.download = `post-${index}.png`;
document.body.appendChild(link);
link.click();
// Clean up
document.body.removeChild(link);
document.body.removeChild(tempContainer);
return image;
} catch (error) {
console.error("Screenshot capture failed:", error);
document.body.removeChild(tempContainer);
throw error;
}
}
$: {
if (selectedSector || $mode) {
config = plotData() || null;
}
}
</script>
<SEO
title="POTUS Tracker: Real-Time Presidential Schedule, Executive Orders & Legislation"
description="Track the President of the United States in real-time. Get updates on the POTUS schedule, executive orders, signed legislation, and official events."
/>
<section
class="w-full max-w-3xl sm:max-w-[1400px] overflow-hidden min-h-screen pb-20 pt-5 px-4 lg:px-3 text-muted dark:text-white"
>
<div class="text-sm sm:text-[1rem] breadcrumbs">
<ul>
<li><a href="/" class="text-muted dark:text-gray-300">Home</a></li>
<li class="text-muted dark:text-gray-300">POTUS Tracker</li>
</ul>
</div>
<div class="w-full flex h-full overflow-hidden">
<div
class="w-full relative flex justify-center items-center overflow-hidden"
>
<div
class="relative flex flex-row justify-center items-start overflow-hidden w-full"
>
<div class="w-full mt-5">
<div class="lg:float-left lg:w-[calc(100%-336px-20px)]">
<div class=" border-b-[2px]">
<h1 class="mb-1 text-2xl sm:text-3xl font-bold">POTUS Tracker</h1>
</div>
</div>
<div class=" lg:float-left lg:w-[calc(100%-336px-40px)]">
<div class="mt-5 mb-5">
<Infobox
text={`Since the inauguration of Donald J. Trump on January 20, 2025, the
${selectedSector} has ${data?.getData?.marketPerformance[sectorDict[selectedSector]]["Inauguration"] >= 0 ? "grown" : "declined"} by <span class="${data?.getData?.marketPerformance[sectorDict[selectedSector]]["Inauguration"] >= 0 ? "text-green-600 dark:text-[#00FC50] before:content-['+']" : "text-red-600 dark:text-[#FF2F1F]"}">
${data?.getData?.marketPerformance[sectorDict[selectedSector]]["Inauguration"] ?? "n/a"}%</span>.`}
/>
</div>
<div class="flex flex-row items-center w-fit ml-auto mt-2 sm:mt-0">
<div class="relative inline-block text-left grow">
<DropdownMenu.Root>
<DropdownMenu.Trigger asChild let:builder>
<Button
builders={[builder]}
class="shadow-sm w-full border-gray-300 dark:border-gray-600 border bg-white sm:hover:bg-gray-100 dark:bg-default dark:sm:hover:bg-primary ease-out flex flex-row justify-between items-center px-3 py-2 rounded-md truncate"
>
<span class="truncate">{selectedSector}</span>
<svg
class="-mr-1 ml-1 h-5 w-5 xs:ml-2 inline-block"
viewBox="0 0 20 20"
fill="currentColor"
style="max-width:40px"
aria-hidden="true"
>
<path
fill-rule="evenodd"
d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z"
clip-rule="evenodd"
></path>
</svg>
</Button>
</DropdownMenu.Trigger>
<DropdownMenu.Content
class="w-56 h-fit max-h-72 overflow-y-auto scroller"
>
<DropdownMenu.Label
class="text-muted dark:text-muted dark:text-gray-300"
>
Select Sector
</DropdownMenu.Label>
<DropdownMenu.Separator />
<DropdownMenu.Group>
{#each updatedSectorList as sector}
{#if sector === "S&P500" || ["Pro", "Plus"]?.includes(data?.user?.tier)}
<DropdownMenu.Item
on:click={() => (selectedSector = sector)}
class="cursor-pointer sm:hover:bg-gray-200 dark:sm:hover:bg-primary"
>
{sector}
</DropdownMenu.Item>
{:else}
<DropdownMenu.Item
on:click={() => goto("/pricing")}
class="cursor-pointer sm:hover:bg-gray-200 dark:sm:hover:bg-primary"
>
{sector}
<svg
class="ml-1 size-4"
viewBox="0 0 20 20"
fill="currentColor"
style="max-width: 40px;"
>
<path
fill-rule="evenodd"
d="M5 9V7a5 5 0 0110 0v2a2 2 0 012 2v5a2 2 0 01-2 2H5a2 2 0 01-2-2v-5a2 2 0 012-2zm8-2v2H7V7a3 3 0 016 0z"
clip-rule="evenodd"
>
</path>
</svg>
</DropdownMenu.Item>
{/if}
{/each}
</DropdownMenu.Group>
</DropdownMenu.Content>
</DropdownMenu.Root>
</div>
</div>
<div
class="chart mt-5 border border-gray-300 dark:border-gray-800 rounded"
use:highcharts={config}
></div>
<nav
class="border-[#2C6288] dark:border-white border-b-[2px] overflow-x-auto whitespace-nowrap no-scrollbar mt-4"
>
<ul class="flex flex-row items-center w-full text-[1rem]">
{#each tabs as item, i}
<button
on:click={() => (activeIdx = i)}
class="p-2 px-5 cursor-pointer {activeIdx === i
? 'text-muted dark:text-white bg-[#EEEEEE] dark:bg-primary/90 font-semibold'
: 'text-blue-500 dark:text-gray-400 sm:hover:text-muted dark:sm:hover:text-white sm:hover:bg-[#EEEEEE] dark:sm:hover:bg-primary/90'}"
>
{item.title}
</button>
{/each}
</ul>
</nav>
{#if activeIdx === 0}
<h3
class=" text-lg sm:text-xl font-semibold mb-2 mt-6 border-y border-gray-300 dark:border-gray-800 pt-2 pb-2"
>
Official Presidential Schedule
</h3>
<div
class="border border-gray-300 dark:border-gray-800 rounded-md p-4"
>
<div class="space-y-4">
{#each Object?.entries(groupedByDate) as [date, items], indexA}
<div class="my-4">
<div
class="border-b border-gray-300 dark:border-gray-600 pb-2 w-full flex flex-row items-center justify-between"
>
<span class="text-[1rem] sm:text-lg font-semibold">
{date}</span
>
{#if items?.at(0)?.changesPercentage}
<div class="ml-auto text-sm">
<span class="inline-block">S&P500</span>
<span
class="{items?.at(0)?.changesPercentage > 0
? "text-green-600 dark:text-[#00FC50] before:content-['+']"
: 'text-red-600 dark:text-[#FF2F1F]'} "
>{items.length > 0
? items?.at(0)?.changesPercentage
: "n/a"}%</span
>
</div>
{/if}
</div>
<!-- Display date -->
<br />
{#each items as item, indexB}
<div class="flex flex-col items-start space-y-1 mb-6">
<div class="flex flex-row items-center space-x-2">
<div class="relative">
<svg
fill={indexA === 0 && indexB === 0
? "#2E86DE"
: "#808080"}
class="w-5 h-5 relative z-10"
viewBox="-51.2 -51.2 614.40 614.40"
id="_78_Circle-Full"
xmlns="http://www.w3.org/2000/svg"
stroke={indexA === 0 && indexB === 0
? "#2E86DE"
: "#808080"}
stroke-width="0.00512"
>
<g id="SVGRepo_bgCarrier" stroke-width="0"></g>
<g
id="SVGRepo_tracerCarrier"
stroke-linecap="round"
stroke-linejoin="round"
stroke="#CCCCCC"
stroke-width="24.576"
></g>
<g id="SVGRepo_iconCarrier">
<path
id="Path_111"
data-name="Path 111"
d="M256,512C114.625,512,0,397.375,0,256S114.625,0,256,0,512,114.625,512,256,397.375,512,256,512Zm0-448C149.969,64,64,149.969,64,256s85.969,192,192,192,192-85.969,192-192S362.031,64,256,64Zm0,320A128,128,0,1,1,384,256,128.006,128.006,0,0,1,256,384Z"
fill-rule="evenodd"
></path>
</g>
</svg>
{#if indexA === 0 && indexB === 0}
<span
class="absolute -inset-1 rounded-full animate-ping w-3 h-3 m-auto bg-blue-400/75"
></span>
{/if}
</div>
<span
class="text-sm sm:text-[1rem] text-muted dark:text-gray-400"
>
{item.time_formatted}
{item.location !== null
? `- ${item?.location}`
: ""}
</span>
</div>
<span class="text-sm sm:text-[1rem] ml-7">
{item.details}
</span>
</div>
{/each}
</div>
{/each}
</div>
</div>
{:else if activeIdx === 1}
<h3
class=" text-lg sm:text-xl font-semibold mb-2 mt-6 border-y border-gray-300 dark:border-gray-800 pt-2 pb-2"
>
Executive Actions
</h3>
<div
class=" border border-gray-300 dark:border-gray-800 rounded-md p-4"
>
<div class="space-y-4">
{#each Object.entries(groupedOrders) as [date, items], indexA}
<div class="my-4">
<div
class="border-b border-gray-300 dark:border-gray-600 pb-2 flex flex-row items-center"
>
<span class="text-[1rem] font-semibold">{date}</span>
{#if latestInfoDate(date)}
<label
class="bg-[#fff] rounded text-black font-semibold text-xs px-2 py-0.5 ml-3 inline-block"
>New</label
>
{/if}
</div>
<br />
{#each items as item, indexB}
<!-- Card container -->
<div
class="{indexB > 0
? 'my-4'
: 'my-1'} p-4 rounded-lg border border-gray-300 dark:border-gray-700 shadow-sm bg-white dark:bg-[#111315]"
>
<!-- Top row: avatar + user info -->
<div class="flex items-start space-x-3">
<a
href="https://truthsocial.com/@realDonaldTrump"
target="_blank"
rel="noopener noreferrer"
class="w-10 h-10 rounded-full shrink-0"
>
<img
class="rounded-full"
src={avatar}
alt="Trump Image"
loading="lazy"
/>
</a>
<div class="flex flex-col items-start w-full">
<h3
class="font-semibold text-gray-900 dark:text-white"
>
<span>Donald J. Trump</span>
</h3>
<h4
class="text-sm text-gray-800 dark:text-gray-400"
>
<div>
{item?.title}
<!-- Sentiment badge -->
<div
class={`mt-2 px-3 py-1 rounded text-white text-xs sm:text-sm w-fit
${
item?.sentiment === "Bullish"
? "bg-emerald-500"
: item?.sentiment === "Bearish"
? "bg-red-600"
: "bg-yellow-500"
}`}
>
{item?.sentiment}
</div>
</div>
</h4>
</div>
</div>
<!-- Description -->
<div class="mt-2 w-full">
<span
class="text-md text-gray-800 dark:text-gray-300"
>
{item?.description?.length > 300
? item?.description?.slice(0, 300) + "..."
: item?.description}
</span>
</div>
<div
class="border-b border-gray-300 dark:border-gray-800 mt-4 mb-4"
></div>
<!-- Source link -->
<div class="flex flex-row items-center w-full">
<a
href={item?.link}
rel="noopener noreferrer"
target="_blank"
>
<svg
class="w-5 h-5 text-gray-600 dark:text-gray-300"
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
><g id="SVGRepo_bgCarrier" stroke-width="0"
></g><g
id="SVGRepo_tracerCarrier"
stroke-linecap="round"
stroke-linejoin="round"
></g><g id="SVGRepo_iconCarrier">
<path
d="M14 12C14 14.7614 11.7614 17 9 17H7C4.23858 17 2 14.7614 2 12C2 9.23858 4.23858 7 7 7H7.5M10 12C10 9.23858 12.2386 7 15 7H17C19.7614 7 22 9.23858 22 12C22 14.7614 19.7614 17 17 17H16.5"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
></path>
</g></svg
>
</a>
<label
for="executivePostModal"
on:click={() => {
postTitle = item?.title;
postContent = item?.description;
postDate = item?.date;
postUrl = item?.link;
}}
class=" cursor-pointer bg-blue-600 text-white rounded px-3 py-1.5 text-sm font-semibold sm:hover:bg-blue-700 ml-auto"
>
Read More
</label>
</div>
</div>
{/each}
</div>
{/each}
</div>
</div>
{:else if activeIdx === 2}
<div
class="flex flex-row items-center mb-2 mt-6 border-y border-gray-300 dark:border-gray-800 pt-2 pb-2"
>
<svg
class="w-7 h-7 rounded-full inline-block"
fill="none"
viewBox="0 0 92 92"
xmlns="http://www.w3.org/2000/svg"
><path d="m0 .438202h91.56v91.56h-91.56z" fill="#5448ee" /><g
fill="#fff"
><path d="m67.9385 54.0751h-11.5057v9.3631h11.5057z" /><path
d="m63.4377 37.8944v-9.4562h-23.4446v34.9084h11.9665v-25.4522z"
/><path d="m24 28.4382h11.4878v9.4539h-11.4878z" /></g
></svg
>
<h3 class="ml-2 text-lg sm:text-xl font-semibold">
Truth Social Posts
</h3>
</div>
<div
class="border border-gray-300 dark:border-gray-800 rounded-md p-4"
>
<div class="">
{#each posts as item, index}
<div
id="post-{index}"
class="{index >= 1
? 'my-4'
: 'my-1'} p-4 rounded-lg border border-gray-100 dark:border-gray-700 shadow-sm bg-gray-100 dark:bg-[#111315]"
>
<div class="flex items-start space-x-3">
<a
href="https://truthsocial.com/@realDonaldTrump"
target="_blank"
rel="noopener noreferrer"
class="w-10 h-10 rounded-full shrink-0"
>
<img
class="rounded-full"
src={avatar}
alt="Trump Image"
loading="lazy"
/>
</a>
<div class="flex flex-col items-start w-full">
<h3
class="font-semibold text-gray-900 dark:text-white"
>
<a
href="https://truthsocial.com/@realDonaldTrump"
target="_blank"
rel="noopener noreferrer"
class="hover:text-blue-500"
>
Donald J. Trump
</a>
</h3>
<h4 class="text-sm text-gray-500 dark:text-gray-400">
<a
href="https://truthsocial.com/@realDonaldTrump"
target="_blank"
rel="noopener noreferrer"
class="hover:text-blue-500"
>
@realDonaldTrump
</a>
</h4>
</div>
</div>
<p class="text-md text-gray-800 dark:text-white mt-2">
{item?.content?.length > 400
? item?.content?.slice(0, 400) + "..."
: item?.content}
</p>
<div
class="border-b border-gray-300 dark:border-gray-800 mt-4"
>
<span
class="text-gray-600 dark:text-gray-300 mb-4 text-sm"
>{item?.date}</span
>
</div>
<div class="flex flex-row items-center mt-4 w-full">
<!--
<label
on:click={() => captureScreenshot(index)}
class="cursor-pointer"
>
<svg
class="w-4 h-4 mr-4 text-gray-400"
viewBox="0 -2 32 32"
version="1.1"
xmlns="http://www.w3.org/2000/svg"
xmlns:xlink="http://www.w3.org/1999/xlink"
xmlns:sketch="http://www.bohemiancoding.com/sketch/ns"
fill="#ffffff"
><g id="SVGRepo_bgCarrier" stroke-width="0"></g><g
id="SVGRepo_tracerCarrier"
stroke-linecap="round"
stroke-linejoin="round"
></g><g id="SVGRepo_iconCarrier">
<title>camera</title>
<desc>Created with Sketch Beta.</desc>
<defs> </defs>
<g
id="Page-1"
stroke="none"
stroke-width="1"
fill="none"
fill-rule="evenodd"
sketch:type="MSPage"
>
<g
id="Icon-Set"
sketch:type="MSLayerGroup"
transform="translate(-256.000000, -465.000000)"
fill="#000000"
>
<path
d="M272,487 C268.687,487 266,484.313 266,481 C266,477.687 268.687,475 272,475 C275.313,475 278,477.687 278,481 C278,484.313 275.313,487 272,487 L272,487 Z M272,473 C267.582,473 264,476.582 264,481 C264,485.418 267.582,489 272,489 C276.418,489 280,485.418 280,481 C280,476.582 276.418,473 272,473 L272,473 Z M286,489 C286,490.104 285.104,491 284,491 L260,491 C258.896,491 258,490.104 258,489 L258,473 C258,471.896 258.896,471 260,471 L264,471 L265,469 C265.707,467.837 265.896,467 267,467 L277,467 C278.104,467 278.293,467.837 279,469 L280,471 L284,471 C285.104,471 286,471.896 286,473 L286,489 L286,489 Z M284,469 L281,469 L280,467 C279.411,465.837 279.104,465 278,465 L266,465 C264.896,465 264.53,465.954 264,467 L263,469 L260,469 C257.791,469 256,470.791 256,473 L256,489 C256,491.209 257.791,493 260,493 L284,493 C286.209,493 288,491.209 288,489 L288,473 C288,470.791 286.209,469 284,469 L284,469 Z"
id="camera"
sketch:type="MSShapeGroup"
>
</path>
</g>
</g>
</g></svg
>
</label>
<label>
<svg
class="w-5 h-5 text-gray-600"
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
><g id="SVGRepo_bgCarrier" stroke-width="0"></g><g
id="SVGRepo_tracerCarrier"
stroke-linecap="round"
stroke-linejoin="round"
></g><g id="SVGRepo_iconCarrier">
<path
d="M14 12C14 14.7614 11.7614 17 9 17H7C4.23858 17 2 14.7614 2 12C2 9.23858 4.23858 7 7 7H7.5M10 12C10 9.23858 12.2386 7 15 7H17C19.7614 7 22 9.23858 22 12C22 14.7614 19.7614 17 17 17H16.5"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
></path>
</g></svg
>
</label>
-->
<label
for="socialPostModal"
on:click={() => {
postContent = item?.content;
postDate = item?.date;
}}
class="cursor-pointer bg-blue-600 text-white rounded px-3 py-1.5 text-sm font-semibold sm:hover:bg-blue-700 ml-auto"
>
Read More
</label>
</div>
</div>
{/each}
</div>
</div>
{/if}
</div>
<div class="order-4 shrink-0 lg:float-right lg:w-[336px]">
<div
class="w-full border border-gray-300 dark:border-gray-600 rounded-md h-fit pb-4 mt-4 cursor-pointer sm:hover:shadow-lg dark:sm:hover:bg-secondary transition ease-out duration-100"
>
<a
href={`/newsletter`}
class="w-auto lg:w-full p-1 flex flex-col m-auto px-2 sm:px-0"
>
<div class="w-full flex justify-between items-center p-3 mt-3">
<h2 class="text-start text-xl font-semibold ml-3">
Market Newsletter
</h2>
</div>
<span class=" p-3 ml-3 mr-3">
Get a daily email with the top market news in bullet point
format.
</span>
</a>
</div>
<div
class="w-full border border-gray-300 dark:border-gray-600 rounded-md h-fit pb-4 mt-4 cursor-pointer sm:hover:shadow-lg dark:sm:hover:bg-secondary transition ease-out duration-100"
>
<a
href={"/stock-screener"}
class="w-auto lg:w-full p-1 flex flex-col m-auto px-2 sm:px-0"
>
<div class="w-full flex justify-between items-center p-3 mt-3">
<h2 class="text-start text-xl font-semibold ml-3">
Stock Screener
</h2>
</div>
<span class=" p-3 ml-3 mr-3">
Build your Stock Screener to find profitable stocks.
</span>
</a>
</div>
<div
class="w-full border border-gray-300 dark:border-gray-600 rounded-md h-fit pb-4 mt-4 cursor-pointer sm:hover:shadow-lg dark:sm:hover:bg-secondary transition ease-out duration-100"
>
<a
href={"/watchlist/stocks"}
class="w-auto lg:w-full p-1 flex flex-col m-auto px-2 sm:px-0"
>
<div class="w-full flex justify-between items-center p-3 mt-3">
<h2 class="text-start text-xl font-semibold ml-3">
Watchlist
</h2>
</div>
<span class=" p-3 ml-3 mr-3">
Keep track of your favorite stocks in real-time.
</span>
</a>
</div>
</div>
</div>
</div>
</div>
</div>
</section>
<input type="checkbox" id="executivePostModal" class="modal-toggle" />
<dialog id="executivePostModal" class="modal modal-bottom sm:modal-middle">
<label
for="executivePostModal"
class="cursor-pointer modal-backdrop bg-[#000]/40"
></label>
<div
class="modal-box w-full max-w-sm p-6 rounded-lg shadow-lg border
bg-white dark:bg-secondary border border-gray-600 dark:border-gray-800"
style="opacity: 1; transform: none;"
>
<div class="flex items-start space-x-3">
<span class="w-10 h-10 rounded-full shrink-0">
<img
class="rounded-full"
src={avatar}
alt="Trump Image"
loading="lazy"
/>
</span>
<div class="flex flex-col items-start w-full">
<h3 class="font-semibold text-gray-900 dark:text-white">
<a
href="https://truthsocial.com/@realDonaldTrump"
target="_blank"
rel="noopener noreferrer"
class="hover:text-blue-500"
>
Donald J. Trump
</a>
</h3>
<h4 class="text-sm text-gray-800 dark:text-gray-400">{postTitle}</h4>
</div>
</div>
<p class="text-sm sm:text-[1rem] mb-4 mt-4">
{postContent}
</p>
<div class="border-b border-gray-300 dark:border-gray-600">
<span class="text-gray-600 dark:text-gray-300 mb-4 text-sm"
>{new Date(postDate ?? null)?.toLocaleString("en-US", {
month: "long",
day: "numeric",
year: "numeric",
})}</span
>
</div>
<div class="flex justify-end space-x-3 mt-5">
<label
for="executivePostModal"
class="cursor-pointer px-4 py-1.5 rounded text-sm font-medium
bg-blue-600 text-white sm:hover:bg-blue-700"
tabindex="0">Close</label
>
<a
href={postUrl}
rel="noopener noreferrer"
target="_blank"
class="cursor-pointer px-4 py-1.5 rounded text-sm font-medium
bg-blue-600 text-white sm:hover:bg-blue-700"
tabindex="0">Read Source</a
>
</div>
</div>
</dialog>
<input type="checkbox" id="socialPostModal" class="modal-toggle" />
<dialog id="socialPostModal" class="modal modal-bottom sm:modal-middle">
<label
for="socialPostModal"
class="cursor-pointer modal-backdrop bg-[#000]/40"
></label>
<div
class="modal-box w-full max-w-sm p-6 rounded-lg shadow-lg border
bg-white dark:bg-secondary border border-gray-600 dark:border-gray-800"
style="opacity: 1; transform: none;"
>
<div class="flex items-start space-x-3">
<a
href="https://truthsocial.com/@realDonaldTrump"
target="_blank"
rel="noopener noreferrer"
class="w-10 h-10 rounded-full shrink-0"
>
<img
class="rounded-full"
src={avatar}
alt="Trump Image"
loading="lazy"
/>
</a>
<div class="flex flex-col items-start w-full">
<h3 class="font-semibold text-gray-900 dark:text-white">
<a
href="https://truthsocial.com/@realDonaldTrump"
target="_blank"
rel="noopener noreferrer"
class="hover:text-blue-500"
>
Donald J. Trump
</a>
</h3>
<h4 class="text-sm text-gray-500 dark:text-gray-400">
<a
href="https://truthsocial.com/@realDonaldTrump"
target="_blank"
rel="noopener noreferrer"
class="hover:text-blue-500"
>
@realDonaldTrump
</a>
</h4>
</div>
</div>
<p class="text-sm sm:text-[1rem] mb-4 mt-4">
{postContent}
</p>
<div class="border-b border-gray-300 dark:border-gray-600">
<span class="text-gray-600 dark:text-gray-300 mb-4 text-sm"
>{postDate}</span
>
</div>
<div class="flex justify-end space-x-3 mt-5">
<label
for="socialPostModal"
class="cursor-pointer px-4 py-1.5 rounded text-sm font-medium
bg-blue-600 text-white sm:hover:bg-blue-700"
tabindex="0">Close</label
>
</div>
</div>
</dialog>