Skip to content

Commit

Permalink
feat: add market stock
Browse files Browse the repository at this point in the history
  • Loading branch information
Max-Mogilski committed Oct 16, 2023
1 parent c577bc0 commit 3a677cb
Show file tree
Hide file tree
Showing 6 changed files with 23 additions and 43 deletions.
8 changes: 5 additions & 3 deletions src/app/[lang]/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -26,9 +26,11 @@ export default async function Web({ params }: { params: { lang: Locale } }) {

return (
<>
<div className="flex w-full justify-end">
<StockDisplay quotes={homepage.stockDailyQuotes} />
</div>
{homepage.marketStock?.data && (
<div className="flex w-full justify-end">
<StockDisplay quotes={homepage.marketStock?.data} />
</div>
)}

<TrendingArticles locale={params.lang} title={homepage.trendingSectionTitle ?? "Trending articles"} />
{homepage.highlightedArticles && (
Expand Down
41 changes: 12 additions & 29 deletions src/components/StockDisplay/StockDisplay.tsx
Original file line number Diff line number Diff line change
@@ -1,32 +1,9 @@
import { z } from "zod"
import { StockDisplayRenderer } from "./StockDisplayRenderer"

type StockQuoteBase = {
id: string
name: string
}
type ValidStockQuote = StockQuoteBase & { quote: AlphaVantageQuote }

type AlphaVantageQuote = {
"Global Quote": {
"10. change percent": number
}
}

export async function StockDisplay({ quotes }: { quotes: { id: string; name: string; quote?: unknown }[] }) {
const validStockQuotes = quotes
.map((stockQuote) => ({
...stockQuote,
quote: validateQuote(stockQuote.quote),
}))
.filter((stockQuote): stockQuote is ValidStockQuote => stockQuote.quote !== null)
.map(({ quote, ...stockQuoteProps }) => ({
id: stockQuoteProps.id,
name: stockQuoteProps.name,
changePercent: quote["Global Quote"]["10. change percent"],
}))

return <StockDisplayRenderer quotes={validStockQuotes} />
export async function StockDisplay({ quotes }: { quotes: Quote[] }) {
const validQuotes = quotes.map((quote) => validateQuote(quote)) as Quote[]
return <StockDisplayRenderer quotes={validQuotes} />
}

function validateQuote(stockQuote: unknown) {
Expand All @@ -36,7 +13,13 @@ function validateQuote(stockQuote: unknown) {
}

const quoteSchema = z.object({
"Global Quote": z.object({
"10. change percent": z.string().transform((val) => parseFloat(val.slice(0, -1))),
}),
id: z.string(),
name: z.string(),
change: z.string().transform((val) => parseFloat(val.slice(0, -1))),
})

type Quote = {
id: string
name: string
change: number
}
2 changes: 1 addition & 1 deletion src/components/StockDisplay/StockDisplayRenderer.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { StockQuote } from "./StockQuote"

export type Quote = { name: string; id: string; changePercent: number }
export type Quote = { name: string; id: string; change: number }

type StockDisplayRendererProps = {
quotes: Quote[]
Expand Down
4 changes: 2 additions & 2 deletions src/components/StockDisplay/StockQuote.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,13 @@ import { cn } from "@/utils/cn"
import { Quote } from "./StockDisplayRenderer"

export function StockQuote({ quote }: { quote: Quote }) {
const isNegative = quote.changePercent < 0
const isNegative = quote.change < 0
return (
<div className="flex items-center justify-center gap-4 border-r-2 border-slate-100 px-10 text-sm">
<span className="whitespace-nowrap font-medium">{quote.name}</span>
<span className={cn("flex items-center gap-1", isNegative ? "text-red-800" : "text-green-800")}>
{!isNegative && "+"}
{quote.changePercent.toFixed(2)}%
{quote.change.toFixed(2)}%
{isNegative ? <MoveDown className="mt-0.5 h-3 w-3" /> : <MoveUp className="mt-0.5 h-3 w-3" />}
</span>
</div>
Expand Down
5 changes: 2 additions & 3 deletions src/lib/client.ts
Original file line number Diff line number Diff line change
Expand Up @@ -80,13 +80,12 @@ export async function getFooter(locale: Locale) {
}

export async function getHomepage(locale: Locale) {
const { homepages } = await graphqlFetch({
const { homepages, marketStock } = await graphqlFetch({
document: getHomepageQuery,
tags: ["HOMEPAGE", "CATEGORY", "ARTICLE"],
revalidate: 60 * 60 * 12, // 12h
variables: { locale },
})
return homepages[0] ?? null
return { ...homepages[0], marketStock }
}

export async function getHomepageMetadata(locale: Locale) {
Expand Down
6 changes: 1 addition & 5 deletions src/lib/queries/components.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,12 +26,8 @@ export const getNavigationQuery = graphql(`

export const getHomepageQuery = graphql(`
query getHomepage($locales: [Locale!]!) {
marketStock
homepages(locales: $locales, first: 1) {
stockDailyQuotes {
id
name
quote
}
recentSectionTitle
trendingSectionTitle
highlightedCategoryTitle
Expand Down

0 comments on commit 3a677cb

Please sign in to comment.