Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixes #67 #69 #68 #72 #76

Merged
merged 7 commits into from
Dec 29, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions app/root.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -206,8 +206,14 @@ function App() {
<div className="font-light">Equestrian</div>
<div className="font-bold">Volunteer Scheduler</div>
</Link>
<div className="flex items-center justify-start gap-1">
<Link to="/tos" className="text-sm mr-2">Terms of Service</Link>
<Link to="/privacy" className="text-sm">Privacy Policy</Link>
</div>

<ThemeSwitch userPreference={data.requestInfo.session.theme} />
</div>

<div className="h-5" />
<Confetti confetti={data.flash?.confetti} />
<Toaster />
Expand Down
2 changes: 1 addition & 1 deletion app/routes/_auth+/onboarding.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -169,7 +169,7 @@ export default function OnboardingPage() {
<Spacer size="xs" />
<Form method="POST" className="mx-auto w-full max-w-sm" {...form.props}>
<Field
labelProps={{ htmlFor: fields.username.id, children: 'Username' }}
labelProps={{ htmlFor: fields.username.id, children: 'Username (Cannot be an email address)' }}
inputProps={{
...conform.input(fields.username),
autoComplete: 'username',
Expand Down
31 changes: 28 additions & 3 deletions app/routes/_marketing+/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,34 @@ import { horseMountains, ohack } from './logos/logos.ts'
import { Link } from '@remix-run/react'
import { Button } from '~/components/ui/button.tsx'
import { useOptionalUser } from '~/utils/user.ts'
// Get siteName from data.ts
import { siteName } from '~/data.ts'

export const meta: V2_MetaFunction = () => [
{ title: 'Girard Training Stables' },
]

export const meta: V2_MetaFunction = () => {

return [
{
title: siteName,
},
{
property: "og:title",
content: siteName,
},
{
name: "description",
content: `Welcome to ${siteName}, the premier equestrian training facility.`,
},
{
name: "og:description",
content: `Welcome to ${siteName}, the premier equestrian training facility.`,
},
{
name: "og:image",
content: "/img/calendar-icon-with-horse-at-grand-canyon-using-arizona-flag-colors.jpeg",
},
];
};

export default function Index() {
const user = useOptionalUser()
Expand Down Expand Up @@ -80,3 +104,4 @@ export default function Index() {
</main>
)
}

55 changes: 54 additions & 1 deletion app/routes/_marketing+/privacy.tsx
Original file line number Diff line number Diff line change
@@ -1,3 +1,56 @@
import {
type V2_MetaFunction
} from '@remix-run/node'

export const meta: V2_MetaFunction = () => {
return [
{
title: "Privacy Policy | TrotTrack.org",
},
{
property: "og:title",
content: "Privacy Policy | TrotTrack.org",
},
{
name: "description",
content: "Read our Privacy Policy for TrotTrack.org, a nonprofit equestrian volunteer system that connects volunteers with equestrian organizations.",
},
];
};

export default function PrivacyRoute() {
return <div>Privacy</div>
return (
<div className="container flex min-h-full flex-col justify-center pb-32 pt-20">
<div className="mx-auto w-full max-w-lg">
<h1 className="text-2xl font-bold">Privacy Policy</h1>
</div>
<p className="text-lg">By using our website and services, you agree to comply with the following privacy policy:</p>
<section>
<h2 className="text-xl font-bold">1. Information Collection</h2>
<p>We collect certain information when you use our website and services. This may include personal information such as your name, email address, and other contact details. We also collect non-personal information such as your IP address and browsing behavior.</p>
</section>
<section>
<h2 className="text-xl font-bold">2. Use of Information</h2>
<p>We use the information we collect to provide and improve our website and services. We may use your personal information to communicate with you, respond to your inquiries, and send you relevant updates and notifications.</p>
</section>
<section>
<h2 className="text-xl font-bold">3. Information Sharing</h2>
<p>We may share your information with third-party service providers who assist us in operating our website and services. We may also share your information when required by law or to protect our rights and interests.</p>
</section>
<section>
<h2 className="text-xl font-bold">4. Data Security</h2>
<p>We take reasonable measures to protect the security of your information. However, please note that no method of transmission over the internet or electronic storage is completely secure.</p>
</section>
<section>
<h2 className="text-xl font-bold">5. Cookies</h2>
<p>We use cookies to enhance your browsing experience and provide personalized content. You can choose to disable cookies in your browser settings, but please note that some features of our website may not function properly.</p>
</section>
<section>
<h2 className="text-xl font-bold">6. Changes to this Privacy Policy</h2>
<p>We may update this privacy policy from time to time. Any changes will be posted on this page, and the revised policy will be effective immediately upon posting.</p>
<p>By using our website and services, you agree to this privacy policy. If you do not agree with any part of this policy, please do not use our website or services.</p>
</section>
</div>
);
}

62 changes: 61 additions & 1 deletion app/routes/_marketing+/tos.tsx
Original file line number Diff line number Diff line change
@@ -1,3 +1,63 @@
import React from 'react';
import {
type V2_MetaFunction
} from '@remix-run/node'

export const meta: V2_MetaFunction = () => {
return [
{
title: "Terms of Service | TrotTrack.org",
},
{
property: "og:title",
content: "Terms of Service | TrotTrack.org",
},
{
name: "description",
content: "Read or Terms of Service for TrotTrack.org, a nonprofit equestrian volunteer system that connects volunteers with equestrian organizations.",
},
];
};

export default function TermsOfServiceRoute() {
return <div>Terms of service</div>
return (
<div className="container flex min-h-full flex-col justify-center pb-32 pt-20">
<div className="mx-auto w-full max-w-lg">
<h1 className="text-2xl font-bold">Terms of Service</h1>
</div>
<p className="text-lg">By using our website and services, you agree to comply with the following terms and conditions:</p>
<section>
<h2 className="text-xl font-bold">1. Acceptance of Terms</h2>
<p>These terms of service govern your use of TrotTrack.org. By accessing or using our website, you acknowledge that you have read, understood, and agree to be bound by these terms.</p>
</section>
<section>
<h2 className="text-xl font-bold">2. Description of Services</h2>
<p>TrotTrack.org is a nonprofit equestrian volunteer system that connects volunteers with equestrian organizations. Our services include providing a platform for volunteers to find and sign up for volunteer opportunities, and for organizations to manage their volunteer programs.</p>
</section>
<section>
<h2 className="text-xl font-bold">3. User Responsibilities</h2>
<p>As a user of TrotTrack.org, you agree to:</p>
<ul>
<li>Provide accurate and complete information when creating an account</li>
<li>Use the website and services in compliance with applicable laws and regulations</li>
<li>Respect the privacy and rights of other users</li>
<li>Not engage in any activity that may disrupt or interfere with the proper functioning of TrotTrack.org</li>
</ul>
</section>
<section>
<h2 className="text-xl font-bold">4. Intellectual Property</h2>
<p>All content and materials on TrotTrack.org, including but not limited to text, graphics, logos, images, and software, are the property of TrotTrack.org or its licensors and are protected by intellectual property laws. You may not reproduce, distribute, modify, or create derivative works of any content without prior written consent from TrotTrack.org.</p>
</section>
<section>
<h2 className="text-xl font-bold">5. Limitation of Liability</h2>
<p>TrotTrack.org and its affiliates shall not be liable for any direct, indirect, incidental, special, or consequential damages arising out of or in connection with your use of the website or services.</p>
</section>
<section>
<h2 className="text-xl font-bold">6. Governing Law</h2>
<p>These terms of service shall be governed by and construed in accordance with the laws of the jurisdiction in which TrotTrack.org operates.</p>
<p>By using TrotTrack.org, you agree to these terms of service. If you do not agree with any part of these terms, please do not use our website or services.</p>
</section>
</div>
);
}

17 changes: 17 additions & 0 deletions app/routes/resources+/event-register.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import { json, type DataFunctionArgs } from '~/remix.ts'
import { prisma } from '~/utils/db.server.ts'
import { sendEmail } from '~/utils/email.server.ts'
import { RegistrationEmail, RegistrationNoticeForAdmins } from './registration-emails.server.tsx'
import { UnregistrationEmail } from './unregistration-emails.server.tsx'
import { createEvent, type DateArray } from 'ics'
import type { User, Event } from '@prisma/client'
import { differenceInMinutes } from 'date-fns'
Expand Down Expand Up @@ -64,6 +65,22 @@ export async function action({ request }: DataFunctionArgs) {
},
},
})

sendEmail({
to: user.email,
subject: `Event Unregistration Notification`,
attachments: [],
react: <UnregistrationEmail event={event} role={submission.value.role} />,
}).then(result => {
if (result.status == 'error') {
// TODO: think through this case and how to handle it properly
console.error(
'There was an error sending an event registration email: ',
JSON.stringify(result.error),
)
}
})

notifyAdmins({
user: user,
event: event,
Expand Down
44 changes: 44 additions & 0 deletions app/routes/resources+/unregistration-emails.server.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import type { Event } from '@prisma/client'
import { Container, Html, Tailwind, Text } from '@react-email/components'
import tailwindConfig from '~/../tailwind.config.ts'
import { format } from 'date-fns'
import { volunteerTypes } from '~/data.ts'

export function UnregistrationEmail({
event,
role,
}: {
event: Event
role: 'cleaningCrew' | 'lessonAssistants' | 'sideWalkers' | 'horseLeaders'
}) {
let roleName
for (let v of volunteerTypes) {
if (v.field == role) {
roleName = v.displayName
}
}

return (
<Tailwind config={tailwindConfig}>
<Html lang="en" dir="ltr">
<Container>
<h1>
<Text>Thanks for considering volunteering! We hope to see you again!</Text>
</h1>
<p>
<Text>
You've unregistered to volunteer{' '}
{roleName ? `as one of the ${roleName}` : ''} to help with the
following event:
</Text>
<Text>{event.title}</Text>
<Text>On: {format(event.start, 'MMMM do, y')}</Text>
<Text>
From: {format(event.start, 'p')} - {format(event.end, 'p')}
</Text>
</p>
</Container>
</Html>
</Tailwind>
)
}
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified public/img/horse.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified public/img/user.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.