Compare commits

12 Commits

Author SHA1 Message Date
5a33344ed9 fix(deps): update dependency lucide-react to ^0.526.0
All checks were successful
Check / Lint and Check (push) Successful in 42s
2025-07-26 20:01:19 +00:00
2a371cbaaf fix(deps): update dependency react-day-picker to v9.8.1
All checks were successful
Check / Lint and Check (push) Successful in 56s
2025-07-26 14:01:12 +00:00
17a8e8b422 fix(deps): update dependency react-hook-form to v7.61.1
All checks were successful
Check / Lint and Check (push) Successful in 54s
2025-07-26 03:01:52 +00:00
f844b9da61 fix(deps): update dependency @hookform/resolvers to v5.2.0
All checks were successful
Check / Lint and Check (push) Successful in 49s
2025-07-26 02:01:49 +00:00
aec2d14a90 chore(deps): update dependency eslint to v9.32.0
All checks were successful
Check / Lint and Check (push) Successful in 46s
2025-07-26 01:01:50 +00:00
91f2a8e316 fix(deps): update dependency zod to v4.0.10
All checks were successful
Check / Lint and Check (push) Successful in 1m23s
2025-07-26 00:01:08 +00:00
a6245ea2f4 chore(deps): update dependency turbo to v2.5.5
All checks were successful
Check / Lint and Check (push) Successful in 52s
2025-07-19 02:01:14 +00:00
69d435b8e2 chore(deps): update dependency @types/node to v22.16.5
All checks were successful
Check / Lint and Check (push) Successful in 58s
2025-07-19 01:01:29 +00:00
5f206b8149 chore(deps): update dependency eslint-config-prettier to v10.1.8
All checks were successful
Check / Lint and Check (push) Successful in 53s
2025-07-19 00:01:09 +00:00
bd2ddd9bb6 fix misc issues, upgrade
All checks were successful
Check / Lint and Check (push) Successful in 1m38s
2025-07-15 17:33:11 +02:00
358dc77e5a disable sign-up on event day
Some checks failed
Check / Lint and Check (push) Failing after 30s
2025-07-15 16:28:02 +02:00
1108a66378 nextjs upgrade 2025-07-15 15:12:17 +02:00
9 changed files with 1103 additions and 796 deletions

44
EVENT_DATES_GUIDE.md Normal file
View File

@@ -0,0 +1,44 @@
# Event Dates Management Guide
## How to Block Sign-ups on Event Days
The sign-up form automatically closes at 3pm on specified event dates to prevent last-minute registrations.
### Managing Event Dates
1. Open the `event-dates.json` file in the project root
2. Add or remove dates in the `eventDates` array
3. Use the format `YYYY-MM-DD` (e.g., "2024-12-25" for December 25, 2024)
### Example Configuration
```json
{
"eventDates": ["2025-09-05", "1999-01-01"],
"cutoffTime": "15:00",
"message": "Sign-ups are closed for today's event. Please come back tomorrow."
}
```
### Important Notes
- The cutoff time is set to 3pm (15:00) by default
- Sign-ups will automatically reopen at midnight after an event day
- Users will see a friendly message when sign-ups are closed
- The time zone follows the server's local time
### Adding New Event Dates
Simply add a new date to the array:
```json
"eventDates": [
"2024-12-25",
"2024-12-31",
"2025-01-15",
"2025-02-14",
"2025-03-20" // <- New date added
]
```
Remember to save the file after making changes!

View File

@@ -14,6 +14,7 @@ import { Popover, PopoverContent, PopoverTrigger } from "@/components/ui/popover
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
import { signupFormSubmit } from "@/lib/actions"; import { signupFormSubmit } from "@/lib/actions";
import { useState } from "react"; import { useState } from "react";
import { isSignupBlocked } from "@/lib/signup-time-check";
export const signupFormSchema = z.object({ export const signupFormSchema = z.object({
name: z.string().min(2, { error: "Name is required" }).max(50, { error: "Name is too long" }), name: z.string().min(2, { error: "Name is required" }).max(50, { error: "Name is too long" }),
@@ -26,6 +27,8 @@ export const oldestDate = new Date(new Date().setFullYear(new Date().getFullYear
export default function SignUp() { export default function SignUp() {
const [submitted, setSubmitted] = useState(false); const [submitted, setSubmitted] = useState(false);
const [response, setResponse] = useState<string | null>(null); const [response, setResponse] = useState<string | null>(null);
const signupStatus = isSignupBlocked();
const form = useForm<z.infer<typeof signupFormSchema>>({ const form = useForm<z.infer<typeof signupFormSchema>>({
resolver: zodResolver(signupFormSchema), resolver: zodResolver(signupFormSchema),
defaultValues: { defaultValues: {
@@ -35,6 +38,12 @@ export default function SignUp() {
}, },
}); });
async function onSubmit(values: z.infer<typeof signupFormSchema>) { async function onSubmit(values: z.infer<typeof signupFormSchema>) {
// Double-check signup isn't blocked before submitting
const currentStatus = isSignupBlocked();
if (currentStatus.blocked) {
setResponse(currentStatus.message || "Sign-ups are currently closed.");
return;
}
setSubmitted(true); setSubmitted(true);
setResponse(await signupFormSubmit(values)); setResponse(await signupFormSubmit(values));
} }
@@ -109,12 +118,22 @@ export default function SignUp() {
</FormItem> </FormItem>
)} )}
/> />
<Button type="submit" disabled={submitted}> <Button type="submit" disabled={submitted || signupStatus.blocked}>
Submit Submit
</Button> </Button>
</form> </form>
</Form> </Form>
); );
} }
// If signup is blocked, show the message
if (signupStatus.blocked && !response) {
return (
<div className="rounded-lg border bg-orange-50 p-6 text-center">
<p className="text-lg font-semibold text-orange-900 mb-2">Sign-ups Temporarily Closed</p>
<p className="text-orange-800">{signupStatus.message}</p>
</div>
);
}
return response ?? SignupForm(); return response ?? SignupForm();
} }

View File

@@ -2,23 +2,17 @@ import * as React from "react";
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
export interface InputProps extends React.InputHTMLAttributes<HTMLInputElement> {} function Input({ className, type, ...props }: React.ComponentProps<"input">) {
return (
const Input = React.forwardRef<HTMLInputElement, InputProps>( <input
({ className, type, ...props }, ref) => { type={type}
return ( data-slot="input"
<input className={cn(
type={type} "flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium file:text-foreground placeholder:text-muted-foreground focus-visible:outline-hidden focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50",
className={cn( className
"flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium file:text-foreground placeholder:text-muted-foreground focus-visible:outline-hidden focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50", )}
className {...props}
)} />
ref={ref} );
{...props} }
/>
);
}
);
Input.displayName = "Input";
export { Input }; export { Input };

7
event-dates.json Normal file
View File

@@ -0,0 +1,7 @@
{
"eventDates": ["2025-09-05", "1999-01-01"],
"cutoffTime": "15:00",
"message": "Sign-ups are closed for today's event. Please come back tomorrow.",
"internalComment": "Add event dates in YYYY-MM-DD format. Signups will be disabled after 3pm (15:00) on these dates by default."
}

View File

@@ -29,7 +29,7 @@ async function listmonk(data: listmonkData): Promise<string> {
return "An error occurred or this email is already subscribed."; return "An error occurred or this email is already subscribed.";
} }
return "Thanks for signing up! Please check your email for a confirmation."; return "Thanks for signing up! Please check your email for a confirmation.";
} catch (error) { } catch {
return "An error occurred while trying to sign up. Please try again."; return "An error occurred while trying to sign up. Please try again.";
} }
} }

23
lib/signup-time-check.ts Normal file
View File

@@ -0,0 +1,23 @@
import eventConfig from "@/event-dates.json";
export function isSignupBlocked(): { blocked: boolean; message?: string } {
const now = new Date();
const currentDate = now.toISOString().split("T")[0]; // YYYY-MM-DD format
const currentTime = now.toTimeString().slice(0, 5); // HH:MM format
// Check if today is an event date
const isEventDay = eventConfig.eventDates.includes(currentDate);
if (isEventDay) {
// Check if current time is after the cutoff time (default 15:00 / 3pm)
const cutoffTime = eventConfig.cutoffTime || "15:00";
if (currentTime >= cutoffTime) {
return {
blocked: true,
message: eventConfig.message,
};
}
}
return { blocked: false };
}

View File

@@ -9,39 +9,39 @@
"check": "next lint && npx tsc --noEmit" "check": "next lint && npx tsc --noEmit"
}, },
"dependencies": { "dependencies": {
"@hookform/resolvers": "^5.0.0", "@hookform/resolvers": "^5.1.1",
"@radix-ui/react-label": "^2.1.2", "@radix-ui/react-label": "^2.1.7",
"@radix-ui/react-popover": "^1.1.6", "@radix-ui/react-popover": "^1.1.14",
"@radix-ui/react-select": "^2.1.6", "@radix-ui/react-select": "^2.2.5",
"@radix-ui/react-slot": "^1.1.2", "@radix-ui/react-slot": "^1.2.3",
"class-variance-authority": "^0.7.0", "class-variance-authority": "^0.7.1",
"clsx": "^2.1.1", "clsx": "^2.1.1",
"cssnano": "^7.0.1", "cssnano": "^7.1.0",
"date-fns": "^4.0.0", "date-fns": "^4.1.0",
"lucide-react": "^0.525.0", "lucide-react": "^0.526.0",
"next": "^15.2.1", "next": "^15.4.1",
"next-plausible": "^3.12.0", "next-plausible": "^3.12.4",
"postcss-flexbugs-fixes": "^5.0.2", "postcss-flexbugs-fixes": "^5.0.2",
"postcss-preset-env": "^10.0.0", "postcss-preset-env": "^10.2.4",
"react": "^19.0.0", "react": "^19.1.0",
"react-day-picker": "^9.5.1", "react-day-picker": "^9.8.0",
"react-dom": "^19.0.0", "react-dom": "^19.1.0",
"react-hook-form": "^7.54.2", "react-hook-form": "^7.60.0",
"tailwind-merge": "^3.0.0", "tailwind-merge": "^3.3.1",
"tailwindcss-animate": "^1.0.7", "tailwindcss-animate": "^1.0.7",
"zod": "^4.0.0" "zod": "^4.0.5"
}, },
"devDependencies": { "devDependencies": {
"@tailwindcss/postcss": "4.1.11", "@tailwindcss/postcss": "4.1.11",
"@types/node": "22.16.3", "@types/node": "22.16.5",
"@types/react": "19.1.8", "@types/react": "19.1.8",
"@types/react-dom": "19.1.6", "@types/react-dom": "19.1.6",
"eslint": "9.31.0", "eslint": "9.32.0",
"eslint-config-next": "15.3.5", "eslint-config-next": "15.4.1",
"eslint-config-prettier": "10.1.5", "eslint-config-prettier": "10.1.8",
"postcss": "8.5.6", "postcss": "8.5.6",
"tailwindcss": "4.1.11", "tailwindcss": "4.1.11",
"turbo": "2.5.4", "turbo": "2.5.5",
"typescript": "5.8.3" "typescript": "5.8.3"
}, },
"packageManager": "pnpm@10.13.1" "packageManager": "pnpm@10.13.1"

1722
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@@ -22,6 +22,6 @@
}, },
"target": "ES2023" "target": "ES2023"
}, },
"include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"], "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts", "event-dates.json"],
"exclude": ["node_modules"] "exclude": ["node_modules"]
} }