From 2319ed72db53859c9bd2acafc26e4024cc2c43f4 Mon Sep 17 00:00:00 2001 From: Sol Irvine <sol@zenzen.io> Date: Mon, 14 Aug 2023 14:33:42 +0900 Subject: [PATCH] wip: Saving work --- app/[lang]/favicon.ico | Bin 15086 -> 0 bytes app/[lang]/layout.tsx | 19 +++++++++------- app/[lang]/page.tsx | 7 ++---- app/context/language-context.tsx | 7 +++++- components/grid/three-items.tsx | 18 +++++++-------- components/grid/tile.tsx | 2 +- components/icons/namemark.tsx | 34 +++++++++++++++++++++++++++ components/label.tsx | 24 +++++++++++++------ components/layout/menu/modal.tsx | 38 +++++++++++++++---------------- dictionaries/en.json | 12 +++++++--- dictionaries/ja.json | 12 +++++++--- lib/shopify/fragments/product.ts | 3 +++ lib/shopify/types.ts | 3 +++ tailwind.config.js | 2 +- 14 files changed, 124 insertions(+), 57 deletions(-) delete mode 100644 app/[lang]/favicon.ico create mode 100644 components/icons/namemark.tsx diff --git a/app/[lang]/favicon.ico b/app/[lang]/favicon.ico deleted file mode 100644 index dc7d8431e47ee24c71eebaffe30786769b348353..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 15086 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 Cr`cHm diff --git a/app/[lang]/layout.tsx b/app/[lang]/layout.tsx index 49c86e089..20fbbfd69 100644 --- a/app/[lang]/layout.tsx +++ b/app/[lang]/layout.tsx @@ -4,6 +4,7 @@ import localFont from 'next/font/local'; import { ReactNode, Suspense } from 'react'; import { LanguageProvider } from 'app/context/language-context'; +import { getDictionary } from 'dictionaries'; import './globals.css'; const { TWITTER_CREATOR, TWITTER_SITE, SITE_NAME } = process.env; @@ -38,12 +39,6 @@ const cinzel = localFont({ variable: '--font-cinzel' }); -const mincho = localFont({ - src: '../fonts/A-OTF-A1MinchoStd-Bold.otf', - display: 'swap', - variable: '--font-cinzel' -}); - const alpina = localFont({ src: [ { @@ -60,6 +55,12 @@ const alpina = localFont({ variable: '--font-alpina' }); +const mincho = localFont({ + src: '../fonts/A-OTF-A1MinchoStd-Bold.otf', + display: 'swap', + variable: '--font-mincho' +}); + export async function generateStaticParams() { return i18n.locales.map((locale) => ({ lang: locale })); } @@ -69,13 +70,15 @@ export default async function RootLayout({ params }: { children: ReactNode; - params: { lang: string }; + params: { lang: Locale }; }) { + const dictionary = await getDictionary(params?.lang); + return ( <html lang={params.lang} className={`${cinzel.variable} ${alpina.variable} ${mincho.variable}`}> <body className="bg-dark text-white selection:bg-green-800 selection:text-green-400"> <div className="mx-auto max-w-screen-2xl"> - <LanguageProvider language={params.lang as Locale}> + <LanguageProvider language={params.lang as Locale} dictionary={dictionary}> <Navbar /> <Suspense> <main>{children}</main> diff --git a/app/[lang]/page.tsx b/app/[lang]/page.tsx index 3f6d51aa6..1db45240c 100644 --- a/app/[lang]/page.tsx +++ b/app/[lang]/page.tsx @@ -4,8 +4,7 @@ import Footer from 'components/layout/footer'; import { LanguageControl } from 'components/layout/navbar/language-control'; import type { Locale } from '../../i18n-config'; -import Image from 'next/image'; -import Namemark from 'public/assets/images/namemark.png'; +import LogoNamemark from 'components/icons/namemark'; import { Suspense } from 'react'; export const runtime = 'edge'; @@ -20,15 +19,13 @@ export const metadata = { }; export default async function HomePage({ params: { lang } }: { params: { lang: Locale } }) { - // const dictionary = await getDictionary(lang); - return ( <> <div className="invisible absolute right-40 top-12 md:visible"> <LanguageControl lang={lang} /> </div> <div className="px-6 pb-12 pt-6 md:pb-48 md:pl-6 md:pt-12"> - <Image src={Namemark} alt="suginomori brewery" className="max-w-[260px] md:max-w-[600px]" /> + <LogoNamemark className="w-[260px] fill-current md:w-[600px]" /> </div> <ThreeItemGrid lang={lang} /> <Suspense> diff --git a/app/context/language-context.tsx b/app/context/language-context.tsx index 5f31007d7..fd80929d7 100644 --- a/app/context/language-context.tsx +++ b/app/context/language-context.tsx @@ -6,24 +6,29 @@ import { ReactNode, createContext, useContext, useState } from 'react'; interface IContextProps { currentLanguage?: Locale; setCurrentLanguage: (language: Locale) => void; + currentDictionary?: any; } export const LanguageContext = createContext<IContextProps>({} as IContextProps); export function LanguageProvider({ language, + dictionary, children }: { language: Locale; + dictionary?: any; children: ReactNode | ReactNode[] | string; }) { const [currentLanguage, setCurrentLanguage] = useState<Locale>(language || 'en'); + const [currentDictionary] = useState<any | undefined>(dictionary); return ( <LanguageContext.Provider value={{ currentLanguage, - setCurrentLanguage + setCurrentLanguage, + currentDictionary }} > {children} diff --git a/components/grid/three-items.tsx b/components/grid/three-items.tsx index 04fac5b76..df539ed89 100644 --- a/components/grid/three-items.tsx +++ b/components/grid/three-items.tsx @@ -1,36 +1,35 @@ import clsx from 'clsx'; -import { GridTileImage } from 'components/grid/tile'; import { Locale } from 'i18n-config'; import { getCollectionProducts } from 'lib/shopify'; import type { Product } from 'lib/shopify/types'; import Link from 'next/link'; import Label from '../label'; +import { GridTileImage } from './tile'; function ThreeItemGridItem({ item, priority }: { item: Product; priority?: boolean }) { + const size = item?.variants?.[0]?.selectedOptions?.find((option) => option.name === 'size'); return ( - <div className={clsx('md:col-span-2 md:row-span-1')}> + <div className={clsx('col-span-1 row-span-1 md:col-span-2 md:row-span-1')}> <Link - className="relative block aspect-bottle h-full w-full overflow-hidden bg-black/30" + className="relative block aspect-tall w-full overflow-hidden bg-black/30" href={`/product/${item.handle}`} > <GridTileImage src={item.featuredImage.url} - height={1690} - width={1192} fill sizes={'(min-width: 768px) 33vw, 100vw'} priority={priority} alt={item.title} - className="h-full w-full object-cover" /> </Link> - <div className="font-multingual max-w-sm pt-4"> + <div className="font-multilingual max-w-sm pt-4"> <Label title={item.title as string} amount={item.priceRange.maxVariantPrice.amount} currencyCode={item.priceRange.maxVariantPrice.currencyCode} + size={size?.value} /> - <div className="line-clamp-4 pt-2">{item?.description}</div> + <div className="font-regular line-clamp-4 pt-2">{item?.summary?.value}</div> </div> </div> ); @@ -52,7 +51,8 @@ export async function ThreeItemGrid({ lang }: { lang: Locale }) { return ( <section className={clsx( - 'mx-auto grid max-w-screen-2xl gap-6 px-4 pb-4 md:grid-cols-6', + 'mx-auto grid max-w-screen-2xl gap-6 px-4 pb-4 ', + 'grid-cols-1 md:grid-cols-6', 'grid-rows-3 md:grid-rows-1' )} > diff --git a/components/grid/tile.tsx b/components/grid/tile.tsx index 1b24cb438..4729255bc 100644 --- a/components/grid/tile.tsx +++ b/components/grid/tile.tsx @@ -20,7 +20,7 @@ export function GridTileImage({ {props.src ? ( // eslint-disable-next-line jsx-a11y/alt-text -- `alt` is inherited from `props`, which is being enforced with TypeScript <Image - className={clsx('h-full w-full object-contain', { + className={clsx('h-full w-full object-cover', { 'transition duration-300 ease-in-out hover:scale-105': isInteractive })} {...props} diff --git a/components/icons/namemark.tsx b/components/icons/namemark.tsx new file mode 100644 index 000000000..4eca417a3 --- /dev/null +++ b/components/icons/namemark.tsx @@ -0,0 +1,34 @@ +import clsx from 'clsx'; + +export default function LogoNamemark(props: React.ComponentProps<'svg'>) { + return ( + <svg + xmlns="http://www.w3.org/2000/svg" + xmlnsXlink="http://www.w3.org/1999/xlink" + fill="none" + viewBox="0 0 300 99" + className={clsx(props?.className || 'h-10 w-10 fill-current')} + > + <g clipPath="url(#clip0_1_7)"> + <rect width="300" height="99" fill="url(#pattern0)" /> + </g> + <defs> + <pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1"> + <use + xlinkHref="#image0_1_7" + transform="matrix(0.00102848 0 0 0.0031173 -0.672667 -0.209479)" + /> + </pattern> + <clipPath id="clip0_1_7"> + <rect width="300" height="99" fill="white" /> + </clipPath> + <image + id="image0_1_7" + width="1627" + height="418" + xlinkHref="" + /> + </defs> + </svg> + ); +} diff --git a/components/label.tsx b/components/label.tsx index 287a0fd41..3d92faf1d 100644 --- a/components/label.tsx +++ b/components/label.tsx @@ -4,22 +4,32 @@ import Price from './price'; const Label = ({ title, amount, - currencyCode + currencyCode, + size }: { title: string; amount: string; currencyCode: string; + size?: string; }) => { return ( <div className={clsx('@container/label')}> <div className="font-multilingual flex flex-col space-y-2"> <h3 className="mr-4 line-clamp-2 flex-grow text-3xl">{title}</h3> - <Price - className="flex-none" - amount={amount} - currencyCode={currencyCode} - currencyCodeClassName="hidden @[275px]/label:inline" - /> + <div className="flex flex-row items-center space-x-2"> + <Price + className="flex-none" + amount={amount} + currencyCode={currencyCode} + currencyCodeClassName="hidden @[275px]/label:inline" + /> + {!!size && ( + <> + <div>/</div> + <div>{size}</div> + </> + )} + </div> </div> </div> ); diff --git a/components/layout/menu/modal.tsx b/components/layout/menu/modal.tsx index 7e869f570..78c4d4711 100644 --- a/components/layout/menu/modal.tsx +++ b/components/layout/menu/modal.tsx @@ -9,7 +9,7 @@ import { Fragment, useRef, useState } from 'react'; import { LanguageControl } from '../navbar/language-control'; export function MenuModal() { - const { currentLanguage } = useLanguage(); + const { currentLanguage, currentDictionary } = useLanguage(); let [isOpen, setIsOpen] = useState(false); let closeButtonRef = useRef(null); @@ -43,17 +43,6 @@ export function MenuModal() { > <div className="fixed inset-0 z-20" /> </Transition.Child> - <Transition.Child as={Fragment}> - <div className="fixed right-5 top-6 z-40 px-2 py-1 md:top-11"> - <div className="flex flex-row space-x-6"> - <LanguageControl lang={currentLanguage} /> - - <button ref={closeButtonRef} onClick={close} className=""> - <CloseIcon className="h-10 w-10 stroke-current transition-opacity duration-150 hover:opacity-50" /> - </button> - </div> - </div> - </Transition.Child> <Transition.Child as={Fragment} @@ -66,6 +55,16 @@ export function MenuModal() { > <div className="fixed inset-0 z-30 backdrop-blur-sm"> <Dialog.Panel> + <div className="fixed right-5 top-6 z-40 px-2 py-1 md:top-11"> + <div className="flex flex-row space-x-6"> + <LanguageControl lang={currentLanguage} /> + + <button ref={closeButtonRef} onClick={close} className=""> + <CloseIcon className="h-10 w-10 stroke-current transition-opacity duration-150 hover:opacity-50" /> + </button> + </div> + </div> + <div className="fixed inset-0 grid grid-cols-1 place-content-center bg-dark/80"> <div className="flex flex-row justify-end"> <div className="flex flex-col space-y-4 px-6 text-right"> @@ -74,7 +73,7 @@ export function MenuModal() { href="/products" className="font-serif text-4xl font-normal transition-opacity duration-150 hover:opacity-50" > - products + {currentDictionary?.menu?.products} </Link> </div> @@ -83,7 +82,7 @@ export function MenuModal() { href="/shops" className="font-serif text-4xl font-normal transition-opacity duration-150 hover:opacity-50" > - shop list + {currentDictionary?.menu?.shops} </Link> </div> @@ -92,7 +91,7 @@ export function MenuModal() { href="/about" className="font-serif text-4xl font-normal transition-opacity duration-150 hover:opacity-50" > - about narai + {currentDictionary?.menu?.about} </Link> </div> @@ -101,7 +100,7 @@ export function MenuModal() { href="/bar" className="font-serif text-4xl font-normal transition-opacity duration-150 hover:opacity-50" > - sagyobar + {currentDictionary?.menu?.bar} </Link> </div> @@ -110,7 +109,7 @@ export function MenuModal() { href="/concept" className="font-serif text-4xl font-normal transition-opacity duration-150 hover:opacity-50" > - concept + {currentDictionary?.menu?.concept} </Link> </div> @@ -119,7 +118,7 @@ export function MenuModal() { href="/stories" className="font-serif text-4xl font-normal transition-opacity duration-150 hover:opacity-50" > - stories + {currentDictionary?.menu?.stories} </Link> </div> @@ -128,7 +127,7 @@ export function MenuModal() { href="/company" className="font-serif text-4xl font-normal transition-opacity duration-150 hover:opacity-50" > - company + {currentDictionary?.menu?.company} </Link> </div> @@ -137,6 +136,7 @@ export function MenuModal() { href="/contact" className="font-serif text-4xl font-normal transition-opacity duration-150 hover:opacity-50" > + {currentDictionary?.menu?.contact} contact </Link> </div> diff --git a/dictionaries/en.json b/dictionaries/en.json index 250e9f084..ddb07fdf7 100644 --- a/dictionaries/en.json +++ b/dictionaries/en.json @@ -1,6 +1,12 @@ { - "hello": { - "title": "Hello World", - "description": "This is a description" + "menu": { + "products": "products", + "shops": "shop list", + "about": "about narai", + "bar": "sagyobar", + "concept": "concept", + "stories": "stories", + "company": "company", + "contact": "contact" } } diff --git a/dictionaries/ja.json b/dictionaries/ja.json index 63ac1ad5f..d064af1de 100644 --- a/dictionaries/ja.json +++ b/dictionaries/ja.json @@ -1,6 +1,12 @@ { - "hello": { - "title": "こんにちは", - "description": "これはせつめいですよ" + "menu": { + "products": "商品", + "shops": "取り扱い店", + "about": "naraiについて", + "bar": "sagyobar", + "concept": "コンセプト", + "stories": "ストーリー", + "company": "会社概要", + "contact": "contact" } } diff --git a/lib/shopify/fragments/product.ts b/lib/shopify/fragments/product.ts index be14dedca..0d37d1627 100644 --- a/lib/shopify/fragments/product.ts +++ b/lib/shopify/fragments/product.ts @@ -9,6 +9,9 @@ const productFragment = /* GraphQL */ ` title description descriptionHtml + summary: metafield(namespace: "custom", key: "product_summary") { + value + } options { id name diff --git a/lib/shopify/types.ts b/lib/shopify/types.ts index b51ab3e30..2a7479067 100644 --- a/lib/shopify/types.ts +++ b/lib/shopify/types.ts @@ -115,6 +115,9 @@ export type ShopifyProduct = { title: string; description: string; descriptionHtml: string; + summary: { + value: string; + }; options: ProductOption[]; priceRange: { maxVariantPrice: Money; diff --git a/tailwind.config.js b/tailwind.config.js index b3297fe32..7f686e8df 100644 --- a/tailwind.config.js +++ b/tailwind.config.js @@ -15,7 +15,7 @@ module.exports = { japan: ['var(--font-mincho)', 'sans-serif'] }, aspectRatio: { - bottle: '1.11' + tall: '596 / 845' }, keyframes: { fadeIn: {