Compare commits
	
		
			1 Commits
		
	
	
		
			0.0.17
			...
			7391a09ee2
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 7391a09ee2 | 
| @@ -2,7 +2,6 @@ | |||||||
| node_modules | node_modules | ||||||
|  |  | ||||||
| # Ignore common folders that we do not need | # Ignore common folders that we do not need | ||||||
| .git |  | ||||||
| .next | .next | ||||||
| .github | .github | ||||||
| .vscode | .vscode | ||||||
							
								
								
									
										18
									
								
								.gitea/workflows/container-build.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.gitea/workflows/container-build.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | name: Build Production Container | ||||||
|  | run-name: $Env:GITHUB_REF_TYPE Production Build | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     tags: | ||||||
|  |       - *.*.* | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   Explore-Gitea-Actions: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Check out repository code | ||||||
|  |         uses: actions/checkout@v3 | ||||||
|  |       - name: Deploy | ||||||
|  |         run: | | ||||||
|  |           echo "${{ secrets.CONTAINER_REGISTRY_ACCESS_TOKEN }}" | docker login git.grosinger.net --password-stdin | ||||||
|  |           docker build -t git.grosinger.net/west-sound-hall:$Env:GITHUB_REF_TYPE . | ||||||
|  |           docker push git.grosinger.net/west-sound-hall:$Env:GITHUB_REF_TYPE | ||||||
| @@ -1,32 +0,0 @@ | |||||||
| name: Build Production Image |  | ||||||
| run-name: $GITHUB_REF_TYPE Production Build |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     tags: |  | ||||||
|       - '*.*.*' |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   Build Production Image: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     container: |  | ||||||
|       image: git.grosinger.net/tgrosinger/runner-image:0.0.3 |  | ||||||
|     steps: |  | ||||||
|       - name: Check out repository code |  | ||||||
|         uses: actions/checkout@v3 |  | ||||||
|  |  | ||||||
|       - name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v2 |  | ||||||
|  |  | ||||||
|       - name: Login to Gitea Docker registry |  | ||||||
|         uses: docker/login-action@v2 |  | ||||||
|         with: |  | ||||||
|           registry: git.grosinger.net |  | ||||||
|           username: tgrosinger |  | ||||||
|           password: ${{ secrets.CONTAINER_REGISTRY_ACCESS_TOKEN }} |  | ||||||
|  |  | ||||||
|       - name: Build and push |  | ||||||
|         uses: docker/build-push-action@v4 |  | ||||||
|         with: |  | ||||||
|           context: . |  | ||||||
|           push: true |  | ||||||
|           tags: git.grosinger.net/tgrosinger/west-sound-hall:${{ gitea.ref_name  }} |  | ||||||
							
								
								
									
										23
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,7 +1,6 @@ | |||||||
| FROM docker.io/library/node:18-alpine AS build-env | FROM docker.io/library/node:18-alpine AS build-env | ||||||
|  |  | ||||||
| ENV NODE_ENV production | # Disable telemetry | ||||||
| ENV PORT 3000 |  | ||||||
| ENV NEXT_TELEMETRY_DISABLED 1 | ENV NEXT_TELEMETRY_DISABLED 1 | ||||||
|  |  | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| @@ -16,5 +15,23 @@ RUN npm install | |||||||
| # Copy files. Use dockerignore to avoid copying node_modules | # Copy files. Use dockerignore to avoid copying node_modules | ||||||
| COPY . . | COPY . . | ||||||
|  |  | ||||||
|  | # Build | ||||||
|  | RUN npm run build | ||||||
|  |  | ||||||
|  | # Running the app | ||||||
|  | FROM gcr.io/distroless/nodejs:18 AS runner | ||||||
|  | WORKDIR /app | ||||||
|  |  | ||||||
|  | # Mark as prod, disable telemetry, set port | ||||||
|  | ENV NODE_ENV production | ||||||
|  | ENV PORT 3000 | ||||||
|  | ENV NEXT_TELEMETRY_DISABLED 1 | ||||||
|  |  | ||||||
|  | # Copy from build | ||||||
|  | COPY --from=build-env /app/next.config.mjs ./ | ||||||
|  | COPY --from=build-env /app/public ./public | ||||||
|  | COPY --from=build-env /app/.next ./.next | ||||||
|  | COPY --from=build-env /app/node_modules ./node_modules | ||||||
|  |  | ||||||
| # Run app command | # Run app command | ||||||
| CMD ["/bin/ash", "/app/entrypoint.sh"] | CMD ["./node_modules/next/dist/bin/next", "start"] | ||||||
							
								
								
									
										54
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										54
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,41 +1,10 @@ | |||||||
| # West Sound Hall Website | # Spotlight | ||||||
|  |  | ||||||
| This is the website for the West Sound Hall and Community Club on Orcas Island, WA. | Spotlight is a [Tailwind UI](https://tailwindui.com) site template built using [Tailwind CSS](https://tailwindcss.com) and [Next.js](https://nextjs.org). | ||||||
|  |  | ||||||
| https://westsoundhall.org | ## Getting started | ||||||
|  |  | ||||||
| ## Running | To get started with this template, first install the npm dependencies: | ||||||
|  |  | ||||||
| Pre-build containers are created whenever a version is tagged in this |  | ||||||
| repository. Pull the [latest |  | ||||||
| version](https://git.grosinger.net/tgrosinger/-/packages/container/west-sound-hall/) |  | ||||||
| and run on a server with Docker available. |  | ||||||
|  |  | ||||||
| ```sh |  | ||||||
| docker run -p 3000:3000 git.grosinger.net/tgrosinger/west-sound-hall:0.0.14 |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ## Updating |  | ||||||
|  |  | ||||||
| ### Events on the Homepage |  | ||||||
|  |  | ||||||
| The homepage has a list of upcoming events. This list is created from [`src/app/upcoming-events.json`](https://git.grosinger.net/tgrosinger/west-sound-hall/src/branch/main/src/app/upcoming-events.json). To update the events listed, modify that file, tag a new version, and then update the running container to the latest version. |  | ||||||
|  |  | ||||||
| Events in the past will be automatically hidden from view. |  | ||||||
|  |  | ||||||
| ### News Posts |  | ||||||
|  |  | ||||||
| News posts are written in [`src/app/news`](https://git.grosinger.net/tgrosinger/west-sound-hall/src/branch/main/src/app/news). Each post requires a directory within this folder, and the directory title will become the last segment of the news post URL. |  | ||||||
|  |  | ||||||
| To create a new post, create a new directory in that folder, then within that folder create a `page.mdx`. Use an existing news post as a template by copying its `page.mdx` then modify the author, date, title, description, and the body of the post as needed. |  | ||||||
|  |  | ||||||
| Posts are written in [mdx](https://mdxjs.com/) however for most news posts you can just consider the body of the post to be [standard markdown](https://www.markdownguide.org/basic-syntax/). |  | ||||||
|  |  | ||||||
| Photos can be added in the same directory as the `page.mdx` file. Refer to another news post for an example of how to embed them. |  | ||||||
|  |  | ||||||
| ## Developing |  | ||||||
|  |  | ||||||
| To get started, first install the npm dependencies: |  | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| npm install | npm install | ||||||
| @@ -55,8 +24,19 @@ npm run dev | |||||||
|  |  | ||||||
| Finally, open [http://localhost:3000](http://localhost:3000) in your browser to view the website. | Finally, open [http://localhost:3000](http://localhost:3000) in your browser to view the website. | ||||||
|  |  | ||||||
|  | ## Customizing | ||||||
|  |  | ||||||
|  | You can start editing this template by modifying the files in the `/src` folder. The site will auto-update as you edit these files. | ||||||
|  |  | ||||||
| ## License | ## License | ||||||
|  |  | ||||||
| This site is based off of the Spotlight template from Tailwind, and licensed under the [Tailwind UI license](https://tailwindui.com/license). | This site template is a commercial product and is licensed under the [Tailwind UI license](https://tailwindui.com/license). | ||||||
|  |  | ||||||
| It was purchased by Tony Grosinger. | ## Learn more | ||||||
|  |  | ||||||
|  | To learn more about the technologies used in this site template, see the following resources: | ||||||
|  |  | ||||||
|  | - [Tailwind CSS](https://tailwindcss.com/docs) - the official Tailwind CSS documentation | ||||||
|  | - [Next.js](https://nextjs.org/docs) - the official Next.js documentation | ||||||
|  | - [Headless UI](https://headlessui.dev) - the official Headless UI documentation | ||||||
|  | - [MDX](https://mdxjs.com) - the MDX documentation | ||||||
|   | |||||||
| @@ -1,7 +0,0 @@ | |||||||
| #!/bin/bash |  | ||||||
|  |  | ||||||
| # The build step hard-codes variables from the environment into HTML, so it must |  | ||||||
| # be done with the correct .env.local for runtime. |  | ||||||
|  |  | ||||||
| ./node_modules/next/dist/bin/next build |  | ||||||
| ./node_modules/next/dist/bin/next start |  | ||||||
| @@ -9,7 +9,7 @@ function Article({ article }: { article: BlogPostWithSlug }) { | |||||||
|   return ( |   return ( | ||||||
|     <article className="md:grid md:grid-cols-4 md:items-baseline"> |     <article className="md:grid md:grid-cols-4 md:items-baseline"> | ||||||
|       <Card className="md:col-span-3"> |       <Card className="md:col-span-3"> | ||||||
|         <Card.Title href={`/news/${article.slug}`}> |         <Card.Title href={`/blog/${article.slug}`}> | ||||||
|           {article.title} |           {article.title} | ||||||
|         </Card.Title> |         </Card.Title> | ||||||
|         <Card.Eyebrow |         <Card.Eyebrow | ||||||
| @@ -35,7 +35,7 @@ function Article({ article }: { article: BlogPostWithSlug }) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export const metadata: Metadata = { | export const metadata: Metadata = { | ||||||
|   title: 'Club News', |   title: 'Blog', | ||||||
|   description: 'History, Announcements, and more from the West Sound Hall and Community Club.', |   description: 'History, Announcements, and more from the West Sound Hall and Community Club.', | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @@ -44,7 +44,7 @@ export default async function ArticlesIndex() { | |||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <SimpleLayout |     <SimpleLayout | ||||||
|       title="West Sound Hall News" |       title="West Sound Hall Blog" | ||||||
|       intro="History, Announcements, and more from the West Sound Hall and Community Club." |       intro="History, Announcements, and more from the West Sound Hall and Community Club." | ||||||
|     > |     > | ||||||
|       <div className="md:border-l md:border-zinc-100 md:pl-6 md:dark:border-zinc-700/40"> |       <div className="md:border-l md:border-zinc-100 md:pl-6 md:dark:border-zinc-700/40"> | ||||||
| Before Width: | Height: | Size: 644 KiB After Width: | Height: | Size: 644 KiB | 
| @@ -1,124 +0,0 @@ | |||||||
|  |  | ||||||
| import { type Metadata } from 'next' |  | ||||||
| import Link from 'next/link' |  | ||||||
| import clsx from 'clsx' |  | ||||||
|  |  | ||||||
| import { Container } from '@/components/Container' |  | ||||||
| import { UserPlusIcon, GiftIcon, EnvelopeIcon } from '@heroicons/react/24/solid' |  | ||||||
| import { TableCell, TableHeading, TableLeftHeading } from '@/components/Table' |  | ||||||
|  |  | ||||||
| function SocialLink({ |  | ||||||
|   className, |  | ||||||
|   href, |  | ||||||
|   children, |  | ||||||
|   icon: Icon, |  | ||||||
| }: { |  | ||||||
|   className?: string |  | ||||||
|   href: string |  | ||||||
|   icon: React.ComponentType<{ className?: string }> |  | ||||||
|   children: React.ReactNode |  | ||||||
| }) { |  | ||||||
|   return ( |  | ||||||
|     <li className={clsx(className, 'flex')}> |  | ||||||
|       <Link |  | ||||||
|         href={href} |  | ||||||
|         className="group flex text-sm font-medium text-zinc-800 transition hover:text-teal-500 dark:text-zinc-200 dark:hover:text-teal-500" |  | ||||||
|       > |  | ||||||
|         <Icon className="h-6 w-6 flex-none fill-zinc-500 transition group-hover:fill-teal-500" /> |  | ||||||
|         <span className="ml-4">{children}</span> |  | ||||||
|       </Link> |  | ||||||
|     </li> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const metadata: Metadata = { |  | ||||||
|   title: 'West Sound Community Club - Board of Directory', |  | ||||||
|   description: |  | ||||||
|     'The West Sound Community Club on Orcas Island.', |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| export default function Club() { |  | ||||||
|   return ( |  | ||||||
|     <Container className="mt-16 sm:mt-32"> |  | ||||||
|       <div className="grid grid-cols-1 gap-y-16 lg:grid-cols-2 lg:grid-rows-[auto_1fr] lg:gap-y-12"> |  | ||||||
|         <div className="lg:order-first lg:row-span-2"> |  | ||||||
|           <h1 className="text-4xl font-bold tracking-tight text-zinc-800 dark:text-zinc-100 sm:text-5xl"> |  | ||||||
|             The Board of Directors |  | ||||||
|           </h1> |  | ||||||
|           <div className="mt-6 space-y-7 text-base text-zinc-600 dark:text-zinc-400"> |  | ||||||
|             <p> |  | ||||||
|               Elections for the Board of Directors are held annually at the October member meeting and potluck. |  | ||||||
|             </p> |  | ||||||
|  |  | ||||||
|             <p> |  | ||||||
|               If you are interested in being one the ballot at the upcoming |  | ||||||
|               election, please |  | ||||||
|               <a href="mailto:board@westsoundhall.org" |  | ||||||
|                 className="pl-1 text-blue-600 visited:text-purple-600 hover:underline"> |  | ||||||
|                 contact the board |  | ||||||
|               </a>. |  | ||||||
|             </p> |  | ||||||
|           </div> |  | ||||||
|  |  | ||||||
|           <div className="overflow-x-auto -mx-4 sm:-mx-0"> |  | ||||||
|             <div className="inline-block min-w-full py-2 align-middle"> |  | ||||||
|               <table className="min-w-full divide-y divide-gray-300"> |  | ||||||
|                 <thead> |  | ||||||
|                   <tr> |  | ||||||
|                     <th scope="col" className="py-3.5 pl-4 pr-3 sm:pl-0">Name</th> |  | ||||||
|                     <TableHeading>Position</TableHeading> |  | ||||||
|                   </tr> |  | ||||||
|                 </thead> |  | ||||||
|                 <tbody className="divide-y divide-gray-200 "> |  | ||||||
|                   <tr> |  | ||||||
|                     <TableLeftHeading>Lisa Pedersen</TableLeftHeading> |  | ||||||
|                     <TableCell>President</TableCell> |  | ||||||
|                   </tr> |  | ||||||
|                   <tr> |  | ||||||
|                     <TableLeftHeading>Betsy Wareham</TableLeftHeading> |  | ||||||
|                     <TableCell>Vice President</TableCell> |  | ||||||
|                   </tr> |  | ||||||
|                   <tr> |  | ||||||
|                     <TableLeftHeading>Tony Grosinger</TableLeftHeading> |  | ||||||
|                     <TableCell>Secretary</TableCell> |  | ||||||
|                   </tr> |  | ||||||
|                   <tr> |  | ||||||
|                     <TableLeftHeading>Temporarily performed by Secretary</TableLeftHeading> |  | ||||||
|                     <TableCell>Treasurer</TableCell> |  | ||||||
|                   </tr> |  | ||||||
|                   <tr> |  | ||||||
|                     <TableLeftHeading>Mark Gasser</TableLeftHeading> |  | ||||||
|                     <TableCell>Director</TableCell> |  | ||||||
|                   </tr> |  | ||||||
|                   <tr> |  | ||||||
|                     <TableLeftHeading>Leslie Brown</TableLeftHeading> |  | ||||||
|                     <TableCell>Director</TableCell> |  | ||||||
|                   </tr> |  | ||||||
|                 </tbody> |  | ||||||
|               </table> |  | ||||||
|             </div> |  | ||||||
|           </div> |  | ||||||
|         </div> |  | ||||||
|         <div className="lg:pl-20"> |  | ||||||
|           <ul role="list"> |  | ||||||
|             <SocialLink |  | ||||||
|               href="mailto:contact@westsoundhall.org" |  | ||||||
|               icon={EnvelopeIcon} |  | ||||||
|               className="mt-4 border-zinc-100 dark:border-zinc-700/40" |  | ||||||
|             > |  | ||||||
|               contact@westsoundhall.org |  | ||||||
|             </SocialLink> |  | ||||||
|             <SocialLink |  | ||||||
|               href="mailto:contact@westsoundhall.org" |  | ||||||
|               icon={EnvelopeIcon} |  | ||||||
|               className="mt-4 border-zinc-100 dark:border-zinc-700/40" |  | ||||||
|             > |  | ||||||
|               board@westsoundhall.org |  | ||||||
|             </SocialLink> |  | ||||||
|           </ul> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </Container> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
| @@ -2,12 +2,12 @@ | |||||||
| import { type Metadata } from 'next' | import { type Metadata } from 'next' | ||||||
| import Image from 'next/image' | import Image from 'next/image' | ||||||
| import Link from 'next/link' | import Link from 'next/link' | ||||||
| import dynamic from "next/dynamic"; |  | ||||||
| import clsx from 'clsx' | import clsx from 'clsx' | ||||||
|  |  | ||||||
| import { Container } from '@/components/Container' | import { Container } from '@/components/Container' | ||||||
| import { UserPlusIcon, GiftIcon, EnvelopeIcon, UserGroupIcon } from '@heroicons/react/24/solid' | import { UserPlusIcon, GiftIcon, EnvelopeIcon } from '@heroicons/react/24/solid' | ||||||
| import interiorEmptyImage from '@/images/photos/interior-empty.jpg' | import interiorEmptyImage from '@/images/photos/interior-empty.jpg' | ||||||
|  | import { ClubPayment } from './payment'; | ||||||
|  |  | ||||||
| function SocialLink({ | function SocialLink({ | ||||||
|   className, |   className, | ||||||
| @@ -34,7 +34,7 @@ function SocialLink({ | |||||||
| } | } | ||||||
|  |  | ||||||
| export const metadata: Metadata = { | export const metadata: Metadata = { | ||||||
|   title: 'West Sound Community Club', |   title: 'History', | ||||||
|   description: |   description: | ||||||
|     'The West Sound Community Club on Orcas Island.', |     'The West Sound Community Club on Orcas Island.', | ||||||
| } | } | ||||||
| @@ -43,15 +43,6 @@ export const metadata: Metadata = { | |||||||
| // TODO: Replace interiorEmptyImage with a photo from a potluck | // TODO: Replace interiorEmptyImage with a photo from a potluck | ||||||
|  |  | ||||||
| export default function Club() { | export default function Club() { | ||||||
|  |  | ||||||
|   // Dynamic import since ClubPayment uses `document` |  | ||||||
|   const ClubPayment = dynamic( |  | ||||||
|     () => { |  | ||||||
|       return import("./payment"); |  | ||||||
|     }, |  | ||||||
|     { ssr: false } |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <Container className="mt-16 sm:mt-32"> |     <Container className="mt-16 sm:mt-32"> | ||||||
|       <div className="grid grid-cols-1 gap-y-16 lg:grid-cols-2 lg:grid-rows-[auto_1fr] lg:gap-y-12"> |       <div className="grid grid-cols-1 gap-y-16 lg:grid-cols-2 lg:grid-rows-[auto_1fr] lg:gap-y-12"> | ||||||
| @@ -107,15 +98,9 @@ export default function Club() { | |||||||
|             > |             > | ||||||
|               contact@westsoundhall.org |               contact@westsoundhall.org | ||||||
|             </SocialLink> |             </SocialLink> | ||||||
|             <SocialLink href="/board-of-directors" icon={UserGroupIcon} className="mt-4"> |  | ||||||
|               Board of Directors |  | ||||||
|             </SocialLink> |  | ||||||
|           </ul> |           </ul> | ||||||
|         </div> |         </div> | ||||||
|         <div> |         <div> | ||||||
|           <h2 className="text-2xl mb-4 font-semibold leading-6 text-gray-900"> |  | ||||||
|             Join or Renew your Membership |  | ||||||
|           </h2> |  | ||||||
|           <ClubPayment /> |           <ClubPayment /> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| "use client" | "use client" | ||||||
|  |  | ||||||
| import React, { useEffect, useState, FormEvent, useCallback } from 'react'; | import React, { useEffect, useState, FormEvent } from 'react'; | ||||||
| import { AddressElement, Elements } from '@stripe/react-stripe-js'; | import { AddressElement, Elements } from '@stripe/react-stripe-js'; | ||||||
| import { Appearance, StripeAddressElementOptions, StripeElementsOptions, loadStripe } from '@stripe/stripe-js'; | import { Appearance, StripeAddressElementOptions, StripeElementsOptions, loadStripe } from '@stripe/stripe-js'; | ||||||
| import { | import { | ||||||
| @@ -59,6 +59,7 @@ function CheckoutForm({ | |||||||
|   const [selectedAdditionalDonation, setSelectedAdditionalDonation] = useState<number | null>(null); |   const [selectedAdditionalDonation, setSelectedAdditionalDonation] = useState<number | null>(null); | ||||||
|   const [customAmount, setCustomAmount] = useState(''); |   const [customAmount, setCustomAmount] = useState(''); | ||||||
|   const [email, setEmail] = useState(''); |   const [email, setEmail] = useState(''); | ||||||
|  |   const [offsetFees, setOffsetFees] = useState(true); | ||||||
|   const [totalAmount, setTotalAmount] = useState(300); |   const [totalAmount, setTotalAmount] = useState(300); | ||||||
|   const [message, setMessage] = useState<string>(''); |   const [message, setMessage] = useState<string>(''); | ||||||
|   const [isLoading, setIsLoading] = useState(false); |   const [isLoading, setIsLoading] = useState(false); | ||||||
| @@ -110,7 +111,9 @@ function CheckoutForm({ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     if (offsetFees) { | ||||||
|       subtotal = Math.ceil(subtotal * 1.03) |       subtotal = Math.ceil(subtotal * 1.03) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     setTotalAmount(subtotal); |     setTotalAmount(subtotal); | ||||||
|  |  | ||||||
| @@ -125,7 +128,7 @@ function CheckoutForm({ | |||||||
|         }, |         }, | ||||||
|       }), |       }), | ||||||
|     }); |     }); | ||||||
|   }, [paymentIntentID, selectedMembershipLevel, selectedAdditionalDonation, customAmount, email]) |   }, [paymentIntentID, selectedMembershipLevel, selectedAdditionalDonation, customAmount, offsetFees, email]) | ||||||
|  |  | ||||||
|   const handleSubmit = async (e: FormEvent) => { |   const handleSubmit = async (e: FormEvent) => { | ||||||
|     e.preventDefault(); |     e.preventDefault(); | ||||||
| @@ -166,7 +169,7 @@ function CheckoutForm({ | |||||||
|  |  | ||||||
|         {/* Membership Type */} |         {/* Membership Type */} | ||||||
|         <RadioGroup value={selectedMembershipLevel} onChange={setSelectedMembershipLevel} className="space-y-3"> |         <RadioGroup value={selectedMembershipLevel} onChange={setSelectedMembershipLevel} className="space-y-3"> | ||||||
|           <RadioGroup.Label className="text-base font-semibold leading-6 text-gray-900 dark:text-white"> |           <RadioGroup.Label className="text-base font-semibold leading-6 text-gray-900"> | ||||||
|             Select a membership type |             Select a membership type | ||||||
|           </RadioGroup.Label> |           </RadioGroup.Label> | ||||||
|  |  | ||||||
| @@ -177,8 +180,8 @@ function CheckoutForm({ | |||||||
|                 value={membership} |                 value={membership} | ||||||
|                 className={({ active }) => |                 className={({ active }) => | ||||||
|                   classNames( |                   classNames( | ||||||
|                     active ? 'border-indigo-600 ring-2 ring-indigo-600' : 'border-gray-200 dark:border-gray-500', |                     active ? 'border-indigo-600 ring-2 ring-indigo-600' : 'border-gray-300', | ||||||
|                     'relative flex cursor-pointer rounded-lg border bg-white hover:bg-gray-50 hover:dark:bg-zinc-600 dark:bg-zinc-700 p-4 shadow-sm focus:outline-none' |                     'relative flex cursor-pointer rounded-lg border bg-white p-4 shadow-sm focus:outline-none' | ||||||
|                   ) |                   ) | ||||||
|                 } |                 } | ||||||
|               > |               > | ||||||
| @@ -186,7 +189,7 @@ function CheckoutForm({ | |||||||
|                   <> |                   <> | ||||||
|                     <span className="flex flex-1 items-center justify-between "> |                     <span className="flex flex-1 items-center justify-between "> | ||||||
|                       <span className="flex flex-col"> |                       <span className="flex flex-col"> | ||||||
|                         <RadioGroup.Label as="span" className="block text-sm font-medium text-gray-900 dark:text-white"> |                         <RadioGroup.Label as="span" className="block text-sm font-medium text-gray-900"> | ||||||
|                           {membership.title} |                           {membership.title} | ||||||
|                         </RadioGroup.Label> |                         </RadioGroup.Label> | ||||||
|                         <RadioGroup.Description as="span" className="mt-1 flex items-center text-sm text-gray-500"> |                         <RadioGroup.Description as="span" className="mt-1 flex items-center text-sm text-gray-500"> | ||||||
| @@ -194,7 +197,7 @@ function CheckoutForm({ | |||||||
|                         </RadioGroup.Description> |                         </RadioGroup.Description> | ||||||
|                       </span> |                       </span> | ||||||
|                       <RadioGroup.Description as="span" className="ml-8 text-sm font-medium"> |                       <RadioGroup.Description as="span" className="ml-8 text-sm font-medium"> | ||||||
|                         <span className="text-gray-900 dark:text-white">${membership.price}</span> |                         <span className="text-gray-900">${membership.price}</span> | ||||||
|                         <span className="text-gray-500">/yr</span> |                         <span className="text-gray-500">/yr</span> | ||||||
|                       </RadioGroup.Description> |                       </RadioGroup.Description> | ||||||
|                     </span> |                     </span> | ||||||
| @@ -215,7 +218,7 @@ function CheckoutForm({ | |||||||
|  |  | ||||||
|         {/* Additional donation */} |         {/* Additional donation */} | ||||||
|         <RadioGroup value={selectedAdditionalDonation} onChange={setSelectedAdditionalDonation} className="space-y-3"> |         <RadioGroup value={selectedAdditionalDonation} onChange={setSelectedAdditionalDonation} className="space-y-3"> | ||||||
|           <RadioGroup.Label className="text-base font-semibold leading-6 text-gray-900 dark:text-white"> |           <RadioGroup.Label className="text-base font-semibold leading-6 text-gray-900"> | ||||||
|             Additional donation |             Additional donation | ||||||
|           </RadioGroup.Label> |           </RadioGroup.Label> | ||||||
|           <div className="grid grid-cols-3 gap-3 sm:grid-cols-5"> |           <div className="grid grid-cols-3 gap-3 sm:grid-cols-5"> | ||||||
| @@ -228,9 +231,9 @@ function CheckoutForm({ | |||||||
|                     'cursor-pointer focus:outline-none', |                     'cursor-pointer focus:outline-none', | ||||||
|                     option === -1 ? 'col-span-2' : '', |                     option === -1 ? 'col-span-2' : '', | ||||||
|                     checked |                     checked | ||||||
|                       ? 'ring-2 ring-indigo-600' |                       ? 'ring-2 ring-indigo-600 ring-offset-2' | ||||||
|                       : 'ring-1 ring-inset ring-gray-200 dark:ring-gray-500 text-gray-900 dark:text-white hover:bg-gray-50 hover:dark:bg-zinc-600', |                       : 'ring-1 ring-inset ring-gray-300 bg-white text-gray-900 hover:bg-gray-50', | ||||||
|                     'flex items-center justify-center rounded-md py-3 px-3 text-sm font-semibold sm:flex-1 bg-white dark:bg-zinc-700' |                     'flex items-center justify-center rounded-md py-3 px-3 text-sm font-semibold sm:flex-1' | ||||||
|                   ) |                   ) | ||||||
|                 } |                 } | ||||||
|               > |               > | ||||||
| @@ -250,7 +253,7 @@ function CheckoutForm({ | |||||||
|                           min="0" |                           min="0" | ||||||
|                           step="1" |                           step="1" | ||||||
|                           onChange={(e) => setCustomAmount(e.target.value)} |                           onChange={(e) => setCustomAmount(e.target.value)} | ||||||
|                           className="block w-full rounded-md border-0 py-1.5 pl-7 pr-12 text-gray-900 dark:text-white dark:bg-zinc-700 ring-1 ring-inset ring-gray-200 dark:ring-gray-500 placeholder:text-gray-400 focus:ring-1 focus:ring-inset focus:ring-gray-500 sm:text-sm sm:leading-6" |                           className="block w-full rounded-md border-0 py-1.5 pl-7 pr-12 text-gray-900 ring-1 ring-inset ring-gray-300 placeholder:text-gray-400 focus:ring-1 focus:ring-inset focus:ring-gray-500 sm:text-sm sm:leading-6" | ||||||
|                           placeholder="Custom" |                           placeholder="Custom" | ||||||
|                           aria-describedby="price-currency" |                           aria-describedby="price-currency" | ||||||
|                         /> |                         /> | ||||||
| @@ -274,12 +277,12 @@ function CheckoutForm({ | |||||||
|         </RadioGroup> |         </RadioGroup> | ||||||
|  |  | ||||||
|         <div className="space-y-3"> |         <div className="space-y-3"> | ||||||
|           <h2 className="text-base font-semibold leading-6 text-gray-900 dark:text-white"> |           <h2 className="text-base font-semibold leading-6 text-gray-900"> | ||||||
|             About you |             About you | ||||||
|           </h2> |           </h2> | ||||||
|  |  | ||||||
|           <div className="rounded-md mb-3 px-3 pb-1.5 pt-2.5 shadow-sm ring-1 ring-inset ring-gray-200 dark:ring-gray-500 focus-within:ring-2 focus-within:ring-indigo-600 dark:bg-zinc-700 dark:text-white"> |           <div className="rounded-md mb-3 px-3 pb-1.5 pt-2.5 shadow-sm ring-1 ring-inset ring-gray-300 focus-within:ring-2 focus-within:ring-indigo-600"> | ||||||
|             <label htmlFor="name" className="block text-xs font-medium text-gray-900 dark:text-zinc-400"> |             <label htmlFor="name" className="block text-xs font-medium text-gray-900"> | ||||||
|               Email |               Email | ||||||
|             </label> |             </label> | ||||||
|             <input |             <input | ||||||
| @@ -288,7 +291,7 @@ function CheckoutForm({ | |||||||
|               id="email" |               id="email" | ||||||
|               value={email} |               value={email} | ||||||
|               onChange={(e) => setEmail(e.target.value)} |               onChange={(e) => setEmail(e.target.value)} | ||||||
|               className="block w-full border-0 p-0 dark:bg-zinc-700 text-gray-900 dark:text-white placeholder:text-gray-400 focus:ring-0 sm:text-sm sm:leading-6" |               className="block w-full border-0 p-0 text-gray-900 placeholder:text-gray-400 focus:ring-0 sm:text-sm sm:leading-6" | ||||||
|               placeholder="you@example.com" |               placeholder="you@example.com" | ||||||
|             /> |             /> | ||||||
|           </div> |           </div> | ||||||
| @@ -296,20 +299,37 @@ function CheckoutForm({ | |||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|         <div className="space-y-3"> |         <div className="space-y-3"> | ||||||
|           <h2 className="text-base font-semibold leading-6 text-gray-900 dark:text-white"> |           <h2 className="text-base font-semibold leading-6 text-gray-900"> | ||||||
|             Payment |             Payment | ||||||
|           </h2> |           </h2> | ||||||
|           <div className="mt-1 text-sm text-gray-500"> |           <div className="mt-1 text-sm text-gray-500"> | ||||||
|             Credit card fees included. If you would like to avoid these fees or |             If you would like to pay by cash or check, please instead | ||||||
|             to pay by cash or check, please instead <a className="underline |             <a className="underline mx-1" href="/WSCC-Membership-Form.pdf">fill out a paper form</a> | ||||||
|             mx-1" href="/WSCC-Membership-Form.pdf">fill out a paper form</a> and |             and mail to the address on the form. | ||||||
|             mail to the address on the form. |  | ||||||
|           </div> |           </div> | ||||||
|           <PaymentElement id="payment-element" /> |           <PaymentElement id="payment-element" /> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|         {/* TODO: Automatically renew toggle? */} |         {/* TODO: Automatically renew toggle? */} | ||||||
|  |  | ||||||
|  |         <div className="relative flex gap-x-3"> | ||||||
|  |           <div className="flex h-6 items-center"> | ||||||
|  |             <input | ||||||
|  |               id="offsetFees" | ||||||
|  |               name="offsetFees" | ||||||
|  |               type="checkbox" | ||||||
|  |               checked={offsetFees} | ||||||
|  |               onChange={(e) => setOffsetFees(e.target.checked)} | ||||||
|  |               className="h-4 w-4 rounded border-gray-300 text-indigo-600 focus:ring-indigo-600" | ||||||
|  |             /> | ||||||
|  |           </div> | ||||||
|  |           <div className="text-sm leading-6"> | ||||||
|  |             <label htmlFor="offsetFees" className="font-medium text-gray-900"> | ||||||
|  |               Help offset credit card fees | ||||||
|  |             </label> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|         <button |         <button | ||||||
|           className="mt-6 w-full rounded-md flex justify-center border border-transparent bg-indigo-600 px-4 py-2 text-sm font-medium text-white shadow-sm hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-indigo-500 focus:ring-offset-2" |           className="mt-6 w-full rounded-md flex justify-center border border-transparent bg-indigo-600 px-4 py-2 text-sm font-medium text-white shadow-sm hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-indigo-500 focus:ring-offset-2" | ||||||
|           disabled={isLoading || !stripe || !elements} |           disabled={isLoading || !stripe || !elements} | ||||||
| @@ -329,37 +349,9 @@ function CheckoutForm({ | |||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
| const DEFAULT_OPTIONS = { | export function ClubPayment() { | ||||||
|   config: { attributes: true, childList: true, subtree: true }, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| function useMutationObservable(targetEl: Node, cb: MutationCallback, options = DEFAULT_OPTIONS) { |  | ||||||
|   const [observer, setObserver] = useState<MutationObserver | null>(null); |  | ||||||
|  |  | ||||||
|   useEffect(() => { |  | ||||||
|     const obs = new MutationObserver(cb); |  | ||||||
|     setObserver(obs); |  | ||||||
|   }, [cb, options, setObserver]); |  | ||||||
|  |  | ||||||
|   useEffect(() => { |  | ||||||
|     if (!observer) return; |  | ||||||
|     const { config } = options; |  | ||||||
|     observer.observe(targetEl, config); |  | ||||||
|     return () => { |  | ||||||
|       if (observer) { |  | ||||||
|         observer.disconnect(); |  | ||||||
|       } |  | ||||||
|     }; |  | ||||||
|   }, [observer, targetEl, options]); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export default function ClubPayment() { |  | ||||||
|   const [clientSecret, setClientSecret] = useState(''); |   const [clientSecret, setClientSecret] = useState(''); | ||||||
|   const [paymentIntent, setPaymentIntent] = useState(''); |   const [paymentIntent, setPaymentIntent] = useState(''); | ||||||
|  |  | ||||||
|   const htmlEl = document.getElementsByTagName('html')[0]; |  | ||||||
|   const darkTheme = htmlEl.classList.contains("dark"); |  | ||||||
|  |  | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     // Create PaymentIntent as soon as the page loads using our local API |     // Create PaymentIntent as soon as the page loads using our local API | ||||||
|     fetch('api/stripe_intent', { |     fetch('api/stripe_intent', { | ||||||
| @@ -377,36 +369,15 @@ export default function ClubPayment() { | |||||||
|       }); |       }); | ||||||
|   }, []); |   }, []); | ||||||
|  |  | ||||||
|   const styles = getComputedStyle(htmlEl); |  | ||||||
|  |  | ||||||
|   const appearance: Appearance = { |   const appearance: Appearance = { | ||||||
|     theme: 'stripe', |     theme: 'stripe', | ||||||
|     variables: { |  | ||||||
|       colorBackground: styles.getPropertyValue('--stripe-background'), |  | ||||||
|       colorText: styles.getPropertyValue('--stripe-foreground'), |  | ||||||
|     }, |  | ||||||
|     labels: 'floating', |     labels: 'floating', | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   const cb = useCallback( |  | ||||||
|     () => { |  | ||||||
|       const updatedHtmlEl = document.getElementsByTagName('html')[0]; |  | ||||||
|       const updatedDarkTheme = updatedHtmlEl.classList.contains("dark"); |  | ||||||
|  |  | ||||||
|       if (updatedDarkTheme !== darkTheme) { |  | ||||||
|         location.reload(); |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     [darkTheme] |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   useMutationObservable(htmlEl, cb); |  | ||||||
|  |  | ||||||
|   const options: StripeElementsOptions = { |   const options: StripeElementsOptions = { | ||||||
|     clientSecret, |     clientSecret, | ||||||
|     appearance, |     appearance | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return <> |   return <> | ||||||
|     {clientSecret && ( |     {clientSecret && ( | ||||||
|       <Elements options={options} |       <Elements options={options} | ||||||
|   | |||||||
| @@ -36,17 +36,17 @@ export async function GET(req: Request) { | |||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   let articleIds = require |   let articleIds = require | ||||||
|     .context('../news', true, /\/page\.mdx$/) |     .context('../blog', true, /\/page\.mdx$/) | ||||||
|     .keys() |     .keys() | ||||||
|     .filter((key) => key.startsWith('./')) |     .filter((key) => key.startsWith('./')) | ||||||
|     .map((key) => key.slice(2).replace(/\/page\.mdx$/, '')); |     .map((key) => key.slice(2).replace(/\/page\.mdx$/, '')); | ||||||
|  |  | ||||||
|   for (let id of articleIds) { |   for (let id of articleIds) { | ||||||
|     let url = String(new URL(`/news/${id}`, req.url)); |     let url = String(new URL(`/blog/${id}`, req.url)); | ||||||
|     let html = await (await fetch(url)).text(); |     let html = await (await fetch(url)).text(); | ||||||
|     let $ = cheerio.load(html); |     let $ = cheerio.load(html); | ||||||
|  |  | ||||||
|     let publicUrl = `${siteUrl}/news/${id}`; |     let publicUrl = `${siteUrl}/blog/${id}`; | ||||||
|     let article = $('article').first(); |     let article = $('article').first(); | ||||||
|     let title = article.find('h1').first().text(); |     let title = article.find('h1').first().text(); | ||||||
|     let date = article.find('time').first().attr('datetime'); |     let date = article.find('time').first().attr('datetime'); | ||||||
|   | |||||||
							
								
								
									
										160
									
								
								src/app/page.tsx
									
									
									
									
									
								
							
							
						
						
									
										160
									
								
								src/app/page.tsx
									
									
									
									
									
								
							| @@ -1,36 +1,22 @@ | |||||||
| import Image from 'next/image' | import Image, { type ImageProps } from 'next/image' | ||||||
|  | import clsx from 'clsx' | ||||||
|  |  | ||||||
| import Link from 'next/link' | import { Button } from '@/components/Button' | ||||||
| import { Card } from '@/components/Card' | import { Card } from '@/components/Card' | ||||||
| import { Container } from '@/components/Container' | import { Container } from '@/components/Container' | ||||||
| import { CalendarDaysIcon } from '@heroicons/react/24/solid' | import { CalendarDaysIcon, EnvelopeIcon } from '@heroicons/react/24/solid' | ||||||
| import exteriorFrontImage from '@/images/photos/exterior-front.png' | import exteriorFrontImage from '@/images/photos/exterior-front.png' | ||||||
|  | import stageImage from '@/images/photos/stage.jpg' | ||||||
|  | import exteriorSoutheastImage from '@/images/photos/exterior-southeast.jpg' | ||||||
|  | import interorEmptyImage from '@/images/photos/interior-empty.jpg' | ||||||
|  | import kitchenImage from '@/images/photos/kitchen.jpg' | ||||||
| import { type BlogPostWithSlug, getAllBlogPosts } from '@/lib/articles' | import { type BlogPostWithSlug, getAllBlogPosts } from '@/lib/articles' | ||||||
| import { formatDate } from '@/lib/formatDate' | import { formatDate } from '@/lib/formatDate' | ||||||
| import { promises as fs } from 'fs'; |  | ||||||
|  |  | ||||||
|  |  | ||||||
| function LinkButton({ |  | ||||||
|   href, |  | ||||||
|   children, |  | ||||||
| }: { |  | ||||||
|   href: string |  | ||||||
|   children: React.ReactNode |  | ||||||
| }) { |  | ||||||
|   return ( |  | ||||||
|     <Link |  | ||||||
|       href={href} |  | ||||||
|       className="rounded-md px-3 py-2 font-semibold text-center transition dark:hover:text-teal-400 bg-sky-300 hover:bg-sky-400 dark:bg-zinc-700/40 dark:hover:bg-zinc-600/40" |  | ||||||
|     > |  | ||||||
|       {children} |  | ||||||
|     </Link> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function Article({ article }: { article: BlogPostWithSlug }) { | function Article({ article }: { article: BlogPostWithSlug }) { | ||||||
|   return ( |   return ( | ||||||
|     <Card as="article"> |     <Card as="article"> | ||||||
|       <Card.Title href={`/news/${article.slug}`}> |       <Card.Title href={`/blog/${article.slug}`}> | ||||||
|         {article.title} |         {article.title} | ||||||
|       </Card.Title> |       </Card.Title> | ||||||
|       <Card.Eyebrow as="time" dateTime={article.date} decorate> |       <Card.Eyebrow as="time" dateTime={article.date} decorate> | ||||||
| @@ -42,6 +28,35 @@ function Article({ article }: { article: BlogPostWithSlug }) { | |||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | function Newsletter() { | ||||||
|  |   return ( | ||||||
|  |     <form | ||||||
|  |       action="/thank-you" | ||||||
|  |       className="rounded-2xl border border-zinc-100 p-6 dark:border-zinc-700/40" | ||||||
|  |     > | ||||||
|  |       <h2 className="flex text-sm font-semibold text-zinc-900 dark:text-zinc-100"> | ||||||
|  |         <EnvelopeIcon className="h-6 w-6 flex-none fill-zinc-100 stroke-zinc-400 dark:fill-zinc-100/10 dark:stroke-zinc-500" /> | ||||||
|  |         <span className="ml-3">Stay up to date</span> | ||||||
|  |       </h2> | ||||||
|  |       <p className="mt-2 text-sm text-zinc-600 dark:text-zinc-400"> | ||||||
|  |         Get notified about upcoming events and stay in touch with the community. | ||||||
|  |       </p> | ||||||
|  |       <div className="mt-6 flex"> | ||||||
|  |         <input | ||||||
|  |           type="email" | ||||||
|  |           placeholder="Email address" | ||||||
|  |           aria-label="Email address" | ||||||
|  |           required | ||||||
|  |           className="min-w-0 flex-auto appearance-none rounded-md border border-zinc-900/10 bg-white px-3 py-[calc(theme(spacing.2)-1px)] shadow-md shadow-zinc-800/5 placeholder:text-zinc-400 focus:border-teal-500 focus:outline-none focus:ring-4 focus:ring-teal-500/10 dark:border-zinc-700 dark:bg-zinc-700/[0.15] dark:text-zinc-200 dark:placeholder:text-zinc-500 dark:focus:border-teal-400 dark:focus:ring-teal-400/10 sm:text-sm" | ||||||
|  |         /> | ||||||
|  |         <Button type="submit" className="ml-4 flex-none"> | ||||||
|  |           Join | ||||||
|  |         </Button> | ||||||
|  |       </div> | ||||||
|  |     </form> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  |  | ||||||
| interface Meeting { | interface Meeting { | ||||||
|   title: string |   title: string | ||||||
|   date: string |   date: string | ||||||
| @@ -84,36 +99,21 @@ function MeetingListItem({ meeting }: { meeting: Meeting }) { | |||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  |  | ||||||
| async function Events() { | function Events() { | ||||||
|   const now = new Date(); |   let events: Array<Meeting> = [ | ||||||
|   const nowYear = now.getFullYear(); |     { | ||||||
|   const nowMonth = now.getMonth() + 1; |       title: 'January Potluck', | ||||||
|   const nowDay = now.getDate(); |       date: '2024-01-20', | ||||||
|  |       startTime: '6:00pm', | ||||||
|   const file = await fs.readFile(process.cwd() + '/src/app/upcoming-events.json', 'utf8'); |     }, | ||||||
|   const allEvents: Array<Meeting> = JSON.parse(file); |     { | ||||||
|  |       title: 'February Potluck', | ||||||
|   // Remove any events in the past. |       date: '2024-02-17', | ||||||
|   const events = allEvents.filter((e) => { |       startTime: '6:00pm', | ||||||
|     const [year, month, day] = e.date.split('-'); |       // endTime: '8:00pm', | ||||||
|  |       // notes: 'Bring your own chair.' | ||||||
|     const parsedYear = parseInt(year) |     }, | ||||||
|     if (parsedYear > nowYear) { |   ] | ||||||
|       return true |  | ||||||
|     } else if (parsedYear < nowYear) { |  | ||||||
|       return false |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const parsedMonth = parseInt(month) |  | ||||||
|     if (parsedMonth > nowMonth) { |  | ||||||
|       return true |  | ||||||
|     } else if (parsedMonth < nowMonth) { |  | ||||||
|       return false |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const parsedDay = parseInt(day) |  | ||||||
|     return parsedDay >= nowDay |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <div className="rounded-2xl border border-zinc-100 p-6 dark:border-zinc-700/40"> |     <div className="rounded-2xl border border-zinc-100 p-6 dark:border-zinc-700/40"> | ||||||
| @@ -135,18 +135,43 @@ async function Events() { | |||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | function Photos() { | ||||||
|  |   let rotations = ['rotate-2', '-rotate-2', 'rotate-2', 'rotate-2', '-rotate-2'] | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div className="mt-16 sm:mt-20"> | ||||||
|  |       <div className="-my-4 flex justify-center gap-5 overflow-hidden py-4 sm:gap-8"> | ||||||
|  |         {[exteriorSoutheastImage, stageImage, exteriorFrontImage, interorEmptyImage, kitchenImage].map((image, imageIndex) => ( | ||||||
|  |           <div | ||||||
|  |             key={image.src} | ||||||
|  |             className={clsx( | ||||||
|  |               'relative aspect-[9/10] w-44 flex-none overflow-hidden rounded-xl bg-zinc-100 dark:bg-zinc-800 sm:w-72 sm:rounded-2xl', | ||||||
|  |               rotations[imageIndex % rotations.length], | ||||||
|  |             )} | ||||||
|  |           > | ||||||
|  |             <Image | ||||||
|  |               src={image} | ||||||
|  |               alt="" | ||||||
|  |               sizes="(min-width: 640px) 18rem, 11rem" | ||||||
|  |               className="absolute inset-0 h-full w-full object-cover" | ||||||
|  |             /> | ||||||
|  |           </div> | ||||||
|  |         ))} | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  |  | ||||||
| export default async function Home() { | export default async function Home() { | ||||||
|   let articles = (await getAllBlogPosts()).slice(0, 4) |   let articles = (await getAllBlogPosts()).slice(0, 4) | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
|       <div className="mx-auto max-w-7xl px-6 mt-24 lg:px-8"> |       <Container className="mt-9"> | ||||||
|         <div className="relative px-4 sm:px-8 lg:px-12"> |         <div className="max-w-2xl"> | ||||||
|           <div className="mx-auto max-w-2xl lg:mx-0 grid lg:max-w-none grid-cols-1 lg:grid-cols-2 lg:gap-x-8 gap-y-10"> |           <h1 className="text-4xl font-bold tracking-tight text-zinc-800 dark:text-zinc-100 sm:text-5xl"> | ||||||
|             <h1 className="text-4xl font-bold tracking-tight text-zinc-800 dark:text-zinc-100 sm:text-6xl lg:col-span-2"> |  | ||||||
|             West Sound Community Hall |             West Sound Community Hall | ||||||
|           </h1> |           </h1> | ||||||
|             <div className="max-w-xl"> |  | ||||||
|           <p className="mt-6 text-base text-zinc-600 dark:text-zinc-400"> |           <p className="mt-6 text-base text-zinc-600 dark:text-zinc-400"> | ||||||
|             The West Sound Community Hall is located in the hamlet of West Sound |             The West Sound Community Hall is located in the hamlet of West Sound | ||||||
|             on Orcas Island, about 10 minutes from the ferry landing and |             on Orcas Island, about 10 minutes from the ferry landing and | ||||||
| @@ -157,20 +182,8 @@ export default async function Home() { | |||||||
|             Facing West Sound, the Hall is at the heart of the West Sound community. |             Facing West Sound, the Hall is at the heart of the West Sound community. | ||||||
|           </p> |           </p> | ||||||
|         </div> |         </div> | ||||||
|             <Image |       </Container> | ||||||
|               src={exteriorFrontImage} |       <Photos /> | ||||||
|               alt="Exterior front of the West Sound Hall" |  | ||||||
|               className="lg:row-span-2 w-full max-w-xl rounded-2xl object-cover lg:max-w-none" |  | ||||||
|             /> |  | ||||||
|             <div className="h-fit max-w-xl rounded-2xl border border-zinc-100 p-6 dark:border-zinc-700/40"> |  | ||||||
|               <div className="flex flex-col gap-y-4"> |  | ||||||
|                 <LinkButton href='/club'>Join or Renew your Membership</LinkButton> |  | ||||||
|                 <LinkButton href='/rental'>Rent the Hall</LinkButton> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|           </div> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|       <Container className="mt-24 md:mt-28"> |       <Container className="mt-24 md:mt-28"> | ||||||
|         <div className="mx-auto grid max-w-xl grid-cols-1 gap-y-20 lg:max-w-none lg:grid-cols-2"> |         <div className="mx-auto grid max-w-xl grid-cols-1 gap-y-20 lg:max-w-none lg:grid-cols-2"> | ||||||
|           <div className="flex flex-col gap-16"> |           <div className="flex flex-col gap-16"> | ||||||
| @@ -178,7 +191,8 @@ export default async function Home() { | |||||||
|               <Article key={article.slug} article={article} /> |               <Article key={article.slug} article={article} /> | ||||||
|             ))} |             ))} | ||||||
|           </div> |           </div> | ||||||
|           <div className="lg:pl-16 xl:pl-24"> |           <div className="space-y-10 lg:pl-16 xl:pl-24"> | ||||||
|  |             <Newsletter /> | ||||||
|             <Events /> |             <Events /> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|   | |||||||
| @@ -6,8 +6,6 @@ import clsx from 'clsx' | |||||||
| import { Container } from '@/components/Container' | import { Container } from '@/components/Container' | ||||||
| import exteriorFront from '@/images/photos/exterior-front.png' | import exteriorFront from '@/images/photos/exterior-front.png' | ||||||
| import { EnvelopeIcon, PencilSquareIcon, QuestionMarkCircleIcon } from '@heroicons/react/24/solid' | import { EnvelopeIcon, PencilSquareIcon, QuestionMarkCircleIcon } from '@heroicons/react/24/solid' | ||||||
| import React from 'react' |  | ||||||
| import { TableCell, TableHeading, TableLeftHeading } from '@/components/Table' |  | ||||||
|  |  | ||||||
| function SocialLink({ | function SocialLink({ | ||||||
|   className, |   className, | ||||||
| @@ -34,9 +32,9 @@ function SocialLink({ | |||||||
| } | } | ||||||
|  |  | ||||||
| export const metadata: Metadata = { | export const metadata: Metadata = { | ||||||
|   title: 'West Sound Hall Rental', |   title: 'History', | ||||||
|   description: |   description: | ||||||
|     'Rental information for the West Sound Hall.', |     'The history of the West Sound Community Hall on Orcas Island.', | ||||||
| } | } | ||||||
|  |  | ||||||
| export default function Rental() { | export default function Rental() { | ||||||
| @@ -109,42 +107,42 @@ export default function Rental() { | |||||||
|               <table className="min-w-full divide-y divide-gray-300"> |               <table className="min-w-full divide-y divide-gray-300"> | ||||||
|                 <thead> |                 <thead> | ||||||
|                   <tr> |                   <tr> | ||||||
|                     <th scope="col" className="py-3.5 pl-4 pr-3 sm:pl-0"></th> |                     <th scope="col" className="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-gray-900 sm:pl-0"></th> | ||||||
|                     <TableHeading>Half day (4 hr)</TableHeading> |                     <th scope="col" className="px-3 py-3.5 text-left text-sm font-semibold text-gray-900">Half day (4 hr)</th> | ||||||
|                     <TableHeading>All day</TableHeading> |                     <th scope="col" className="px-3 py-3.5 text-left text-sm font-semibold text-gray-900">All day</th> | ||||||
|                     <TableHeading>Deposit</TableHeading> |                     <th scope="col" className="px-3 py-3.5 text-left text-sm font-semibold text-gray-900">Deposit</th> | ||||||
|                   </tr> |                   </tr> | ||||||
|                 </thead> |                 </thead> | ||||||
|                 <tbody className="divide-y divide-gray-200 "> |                 <tbody className="divide-y divide-gray-200"> | ||||||
|                   <tr> |                   <tr> | ||||||
|                     <TableLeftHeading>Orcas non-profit organizations</TableLeftHeading> |                     <td className="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-gray-900 sm:pl-0">Orcas non-profit organizations</td> | ||||||
|                     <TableCell>$50 or 2 hours for $30*</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$50 or 2 hours for $30*</td> | ||||||
|                     <TableCell>$100</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$100</td> | ||||||
|                     <TableCell>None†</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">None†</td> | ||||||
|                   </tr> |                   </tr> | ||||||
|                   <tr> |                   <tr> | ||||||
|                     <TableLeftHeading>WSCC and OIYC Members</TableLeftHeading> |                     <td className="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-gray-900 sm:pl-0">WSCC and OIYC Members</td> | ||||||
|                     <TableCell>$50 or 2 hours for $30*</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$50 or 2 hours for $30*</td> | ||||||
|                     <TableCell>$100</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$100</td> | ||||||
|                     <TableCell>None†</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">None†</td> | ||||||
|                   </tr> |                   </tr> | ||||||
|                   <tr> |                   <tr> | ||||||
|                     <TableLeftHeading>Off-island non-profit organizations</TableLeftHeading> |                     <td className="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-gray-900 sm:pl-0">Off-island non-profit organizations</td> | ||||||
|                     <TableCell>$50</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$50</td> | ||||||
|                     <TableCell>$100</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$100</td> | ||||||
|                     <TableCell>$300</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$300</td> | ||||||
|                   </tr> |                   </tr> | ||||||
|                   <tr> |                   <tr> | ||||||
|                     <TableLeftHeading>Individuals and non-public use</TableLeftHeading> |                     <td className="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-gray-900 sm:pl-0">Individuals and non-public use</td> | ||||||
|                     <TableCell>$40 per hour</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$40 per hour</td> | ||||||
|                     <TableCell>$300</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$300</td> | ||||||
|                     <TableCell>$300</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$300</td> | ||||||
|                   </tr> |                   </tr> | ||||||
|                   <tr> |                   <tr> | ||||||
|                     <TableLeftHeading>Government sponsored activities</TableLeftHeading> |                     <td className="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-gray-900 sm:pl-0">Government sponsored activities</td> | ||||||
|                     <TableCell>$50</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$50</td> | ||||||
|                     <TableCell>$50</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">$50</td> | ||||||
|                     <TableCell>None†</TableCell> |                     <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500">None†</td> | ||||||
|                   </tr> |                   </tr> | ||||||
|                 </tbody> |                 </tbody> | ||||||
|               </table> |               </table> | ||||||
| @@ -170,7 +168,6 @@ export default function Rental() { | |||||||
|               The West Sound Community Hall may not be used for either personal or |               The West Sound Community Hall may not be used for either personal or | ||||||
|               organizational monetary gain or to promote business activities. |               organizational monetary gain or to promote business activities. | ||||||
|               The only exceptions to this restriction are: |               The only exceptions to this restriction are: | ||||||
|             </p> |  | ||||||
|  |  | ||||||
|               <ol className="list-decimal ml-8 mt-5"> |               <ol className="list-decimal ml-8 mt-5"> | ||||||
|                 <li> |                 <li> | ||||||
| @@ -186,6 +183,7 @@ export default function Rental() { | |||||||
|                   sponsoring nonprofit organization.) |                   sponsoring nonprofit organization.) | ||||||
|                 </li> |                 </li> | ||||||
|               </ol> |               </ol> | ||||||
|  |             </p> | ||||||
|  |  | ||||||
|             <p> |             <p> | ||||||
|               Any individual or organizational nonpublic use of the hall which |               Any individual or organizational nonpublic use of the hall which | ||||||
|   | |||||||
| @@ -1,24 +0,0 @@ | |||||||
| [ |  | ||||||
|     { |  | ||||||
|         "title": "January Potluck", |  | ||||||
|         "date": "2024-01-20", |  | ||||||
|         "startTime": "6:00pm" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|         "title": "February Potluck", |  | ||||||
|         "date": "2024-02-17", |  | ||||||
|         "startTime": "6:00pm", |  | ||||||
|         "notes": "With historic West Sound slide show." |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|         "title": "Town Hall Meeting", |  | ||||||
|         "date": "2024-03-06", |  | ||||||
|         "startTime": "6:00pm", |  | ||||||
|         "endTime": "8:00pm" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|         "title": "March Potluck", |  | ||||||
|         "date": "2024-03-16", |  | ||||||
|         "startTime": "6:00pm" |  | ||||||
|     } |  | ||||||
| ] |  | ||||||
| @@ -28,18 +28,12 @@ export function Footer() { | |||||||
|             <div className="flex flex-col items-center justify-between gap-6 sm:flex-row"> |             <div className="flex flex-col items-center justify-between gap-6 sm:flex-row"> | ||||||
|               <div className="flex flex-wrap justify-center gap-x-6 gap-y-1 text-sm font-medium text-zinc-800 dark:text-zinc-200"> |               <div className="flex flex-wrap justify-center gap-x-6 gap-y-1 text-sm font-medium text-zinc-800 dark:text-zinc-200"> | ||||||
|                 <NavLink href="/hall-history">History</NavLink> |                 <NavLink href="/hall-history">History</NavLink> | ||||||
|                 <NavLink href="/news">News</NavLink> |  | ||||||
|                 <NavLink href="/rental">Rental</NavLink> |                 <NavLink href="/rental">Rental</NavLink> | ||||||
|                 <NavLink href="/club">Club</NavLink> |                 <NavLink href="/club">Club</NavLink> | ||||||
|               </div> |               </div> | ||||||
|               <div> |  | ||||||
|               <p className="text-sm text-zinc-400 dark:text-zinc-500"> |               <p className="text-sm text-zinc-400 dark:text-zinc-500"> | ||||||
|                 © {new Date().getFullYear()} West Sound Community Club. All rights reserved. |                 © {new Date().getFullYear()} West Sound Community Club. All rights reserved. | ||||||
|               </p> |               </p> | ||||||
|                 <p className="text-sm text-zinc-400 dark:text-zinc-500"> |  | ||||||
|                   WSCC is a 501c3 nonprofit organization - 91-1283768 |  | ||||||
|                 </p> |  | ||||||
|               </div> |  | ||||||
|             </div> |             </div> | ||||||
|           </ContainerInner> |           </ContainerInner> | ||||||
|         </div> |         </div> | ||||||
|   | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,25 +0,0 @@ | |||||||
|  |  | ||||||
| export function TableHeading({ |  | ||||||
|     children |  | ||||||
| }: { |  | ||||||
|     children: React.ReactNode |  | ||||||
| }) { |  | ||||||
|     return <th scope="col" className="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 dark:text-zinc-100">{children}</th> |  | ||||||
|  |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function TableLeftHeading({ |  | ||||||
|     children |  | ||||||
| }: { |  | ||||||
|     children: React.ReactNode |  | ||||||
| }) { |  | ||||||
|     return <td className="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-gray-900 dark:text-zinc-100 sm:pl-0">{children}</td> |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function TableCell({ |  | ||||||
|     children |  | ||||||
| }: { |  | ||||||
|     children: React.ReactNode |  | ||||||
| }) { |  | ||||||
|     return <td className="whitespace-nowrap px-3 py-4 text-sm text-gray-500 dark:text-zinc-400">{children}</td> |  | ||||||
| } |  | ||||||
| @@ -12,7 +12,7 @@ export interface BlogPostWithSlug extends BlogPost { | |||||||
| } | } | ||||||
|  |  | ||||||
| async function importBlogPost(filename: string): Promise<BlogPostWithSlug> { | async function importBlogPost(filename: string): Promise<BlogPostWithSlug> { | ||||||
|   let { article } = (await import(`../app/news/${filename}`)) as { |   let { article } = (await import(`../app/blog/${filename}`)) as { | ||||||
|     default: React.ComponentType; |     default: React.ComponentType; | ||||||
|     article: BlogPost; |     article: BlogPost; | ||||||
|   }; |   }; | ||||||
| @@ -25,7 +25,7 @@ async function importBlogPost(filename: string): Promise<BlogPostWithSlug> { | |||||||
|  |  | ||||||
| export async function getAllBlogPosts() { | export async function getAllBlogPosts() { | ||||||
|   let articleFilenames = await glob('*/page.mdx', { |   let articleFilenames = await glob('*/page.mdx', { | ||||||
|     cwd: './src/app/news', |     cwd: './src/app/blog', | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   const posts = await Promise.all(articleFilenames.map(importBlogPost)); |   const posts = await Promise.all(articleFilenames.map(importBlogPost)); | ||||||
|   | |||||||
| @@ -2,15 +2,3 @@ | |||||||
| @import 'tailwindcss/components'; | @import 'tailwindcss/components'; | ||||||
| @import './prism.css'; | @import './prism.css'; | ||||||
| @import 'tailwindcss/utilities'; | @import 'tailwindcss/utilities'; | ||||||
|  |  | ||||||
| :root { |  | ||||||
|   /* Your default theme */ |  | ||||||
|  |  | ||||||
|   --stripe-background: #FFFFFF; |  | ||||||
|   --stripe-foreground: #000000; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .dark { |  | ||||||
|   --stripe-background: #3f3f46; |  | ||||||
|   --stripe-foreground: #E4E4E7; |  | ||||||
| } |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user