Merge pull request #28 from mcgill-ecsess/office-hour
Office Hours + Cleanup codebase for schemas & $lib
This commit is contained in:
@@ -0,0 +1,42 @@
|
||||
<script lang="ts">
|
||||
import type { OhCMSResponse } from '$lib/schemas';
|
||||
|
||||
function parseTime(timeStr: string): number {
|
||||
let a = timeStr.match(/^(\d{1,2})(?::(\d{2}))?(AM|PM)$/i);
|
||||
if (!a) return 0;
|
||||
|
||||
let hours = parseInt(a[1], 10);
|
||||
let minutes = parseInt(a[2] || '0', 10);
|
||||
let period = a[3];
|
||||
|
||||
if (period.toUpperCase() === 'PM' && hours !== 12) hours += 12;
|
||||
if (period.toUpperCase() === 'AM' && hours === 12) hours = 0;
|
||||
|
||||
return Number(hours * 60 + minutes); // total minutes since midnight
|
||||
}
|
||||
let { allOhs }: { allOhs: OhCMSResponse } = $props();
|
||||
let sortedOHs = $state(
|
||||
allOhs.sort((a, b) => {
|
||||
return parseTime(a.startTime) - parseTime(b.startTime);
|
||||
})
|
||||
);
|
||||
</script>
|
||||
|
||||
<div class="grid min-w-2/3 grid-cols-5 gap-2 place-self-center">
|
||||
{#each ['Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday'] as day}
|
||||
<div>
|
||||
<p>- {day} -</p>
|
||||
{#each sortedOHs.filter((OH) => OH.day == day) as OH}
|
||||
<div
|
||||
class="
|
||||
card w-64 grid h-38 grid-rows-[3fr_2fr_3fr] bg-ecsess-200 text-ecsess-black m-3 place-content-center rounded-xl
|
||||
p-4 text-center"
|
||||
>
|
||||
<p class="text-lg">{OH.startTime} - {OH.endTime}</p>
|
||||
<p class="text-2xl">{OH.member.name.split(" ")[0]}</p>
|
||||
<p class="text-sm italic">{OH.member.position}</p>
|
||||
</div>
|
||||
{/each}
|
||||
</div>
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
44
src/lib/schemas.ts
Normal file
44
src/lib/schemas.ts
Normal file
@@ -0,0 +1,44 @@
|
||||
export type EventPost = {
|
||||
id: string;
|
||||
title: string;
|
||||
description: string;
|
||||
date: string;
|
||||
time: string;
|
||||
location: string;
|
||||
image: string;
|
||||
link: string;
|
||||
category: string;
|
||||
payment: string; // event payment link (e.g., Zeffy)
|
||||
};
|
||||
|
||||
import type { InputValue } from '@portabletext/svelte';
|
||||
|
||||
export type HomepageCMSResponse = {
|
||||
description: InputValue;
|
||||
councilPhoto: string;
|
||||
faqs: {
|
||||
question: string;
|
||||
answer: string;
|
||||
}[];
|
||||
};
|
||||
|
||||
export type OhCMSResponse = {
|
||||
day: string;
|
||||
startTime: string;
|
||||
endTime: string;
|
||||
member: {
|
||||
name: string;
|
||||
position: string;
|
||||
};
|
||||
}[];
|
||||
|
||||
export type CouncilMember = {
|
||||
name: string;
|
||||
email: string;
|
||||
position: string;
|
||||
positionDescription: string;
|
||||
image: string; // URL
|
||||
yearProgram: string;
|
||||
};
|
||||
|
||||
export type Redirect = { shortname: string; url: string };
|
||||
@@ -1,25 +0,0 @@
|
||||
/**
|
||||
* @typedef {Object} EventPost event object
|
||||
* @property {string} id - event id
|
||||
* @property {string} title - event title
|
||||
* @property {string} description - event description
|
||||
* @property {string} date - event date
|
||||
* @property {string} time - event time
|
||||
* @property {string} location - event location
|
||||
* @property {string} image - event image
|
||||
* @property {string} link - event link
|
||||
* @property {string} category - event category
|
||||
* @property {string} payment - event payment link (e.g., Zeffy)3
|
||||
*/
|
||||
|
||||
/**
|
||||
* @typedef {Object} CouncilMember
|
||||
* @property {string} role
|
||||
* @property {string} name
|
||||
* @property {string} email
|
||||
* @property {string} image
|
||||
* @property {string} major
|
||||
* @property {string} year
|
||||
*/
|
||||
|
||||
export {};
|
||||
@@ -1,39 +0,0 @@
|
||||
import { getFromCMS } from 'utils/utils.js';
|
||||
|
||||
const homepageQuery = `*[_type == "homepage"]{
|
||||
"description": description[],
|
||||
"councilPhoto": councilPhoto.asset->url,
|
||||
"faqs": faqs[]{ question, answer },
|
||||
}[0]`;
|
||||
|
||||
const ohQuery = `*[_type=="oh"].schedule[]{
|
||||
day,
|
||||
startTime,
|
||||
endTime,
|
||||
"host": member->name
|
||||
}`;
|
||||
|
||||
export const load = async () => {
|
||||
/**
|
||||
* @description Response data type based on the `homepageQuery` above.
|
||||
* Note that `description` is a rich/portable text type
|
||||
*
|
||||
* @type {{
|
||||
* description: import('@portabletext/svelte').InputValue,
|
||||
* councilPhoto: string,
|
||||
* faqs: [{
|
||||
* question: string,
|
||||
* answer: string
|
||||
* }],
|
||||
* }}
|
||||
*
|
||||
*/
|
||||
let CMSresponse = await getFromCMS(homepageQuery);
|
||||
|
||||
return {
|
||||
description: CMSresponse.description,
|
||||
councilPhoto: CMSresponse.councilPhoto,
|
||||
faqs: CMSresponse.faqs
|
||||
// ohs: await getFromCMS(ohQuery),
|
||||
};
|
||||
};
|
||||
34
src/routes/+page.server.ts
Normal file
34
src/routes/+page.server.ts
Normal file
@@ -0,0 +1,34 @@
|
||||
import { getFromCMS } from '$lib/utils.js';
|
||||
import type { HomepageCMSResponse, OhCMSResponse } from '$lib/schemas';
|
||||
|
||||
const homepageQuery = `*[_type == "homepage"]{
|
||||
"description": description[],
|
||||
"councilPhoto": councilPhoto.asset->url,
|
||||
"faqs": faqs[]{ question, answer },
|
||||
}[0]`;
|
||||
|
||||
const ohQuery = `*[_type=="officeHours"]{
|
||||
day,
|
||||
startTime,
|
||||
endTime,
|
||||
"member": {
|
||||
"name": member->name,
|
||||
"position": member->position
|
||||
}
|
||||
}`;
|
||||
|
||||
export const load = async () => {
|
||||
/**
|
||||
* @description Response data type based on the `homepageQuery` above.
|
||||
* Note that `description` is a rich/portable text type
|
||||
*/
|
||||
let homepageResp: HomepageCMSResponse = await getFromCMS(homepageQuery);
|
||||
let officeHourResp: OhCMSResponse = await getFromCMS(ohQuery);
|
||||
|
||||
return {
|
||||
description: homepageResp.description,
|
||||
councilPhoto: homepageResp.councilPhoto,
|
||||
faqs: homepageResp.faqs,
|
||||
allOHs: officeHourResp
|
||||
};
|
||||
};
|
||||
@@ -2,6 +2,7 @@
|
||||
import FaqAccordion from 'components/FAQAccordion.svelte';
|
||||
import Section from 'components/Section.svelte';
|
||||
import { PortableText } from '@portabletext/svelte';
|
||||
import OhSchedule from 'components/OHSchedule.svelte';
|
||||
|
||||
/** loading things from the server side */
|
||||
let { data } = $props();
|
||||
@@ -46,6 +47,6 @@
|
||||
<Section>
|
||||
<div>
|
||||
<h1 class="text-2xl">Office Hours</h1>
|
||||
<p>Under development</p>
|
||||
<OhSchedule allOhs={data.allOHs}/>
|
||||
</div>
|
||||
</Section>
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
import { getFromCMS } from 'utils/utils.js';
|
||||
import { getFromCMS } from '$lib/utils.js';
|
||||
|
||||
const query = `*[_type == "members"]{
|
||||
name,
|
||||
@@ -34,7 +34,7 @@
|
||||
selectedMember = member;
|
||||
}
|
||||
|
||||
console.log(ureps)
|
||||
console.log(ureps);
|
||||
|
||||
// svelte-ignore state_referenced_locally
|
||||
// console.log(selectedMember);
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
import { getFromCMS } from 'utils/utils.js';
|
||||
import { getFromCMS } from '$lib/utils.js';
|
||||
|
||||
// needs to concat and format this text
|
||||
const eventQuery = `*[_type == "events"]{
|
||||
|
||||
@@ -1,23 +0,0 @@
|
||||
import { redirect } from '@sveltejs/kit';
|
||||
import { getFromCMS } from 'utils/utils.js';
|
||||
|
||||
const redirectQuery = `*[_type == "redirects"]{ shortname, url }`;
|
||||
|
||||
export const load = async ({ params }) => {
|
||||
/** @type {[{shortname: String, url: String}]} */
|
||||
let CMSresponse = await getFromCMS(redirectQuery);
|
||||
|
||||
const { shortname } = params;
|
||||
|
||||
CMSresponse.forEach(res => {
|
||||
if(res.shortname == shortname) {
|
||||
// if match
|
||||
throw redirect(302, res.url);
|
||||
}
|
||||
});
|
||||
|
||||
return {
|
||||
shortname: shortname,
|
||||
availableShortnames: CMSresponse,
|
||||
}
|
||||
};
|
||||
23
src/routes/r/[shortname]/+page.server.ts
Normal file
23
src/routes/r/[shortname]/+page.server.ts
Normal file
@@ -0,0 +1,23 @@
|
||||
import { redirect } from '@sveltejs/kit';
|
||||
import { getFromCMS } from '$lib/utils.js';
|
||||
import type { Redirect } from '$lib/schemas';
|
||||
|
||||
const redirectQuery = `*[_type == "redirects"]{ shortname, url }`;
|
||||
|
||||
export const load = async ({ params }) => {
|
||||
let CMSresponse: Redirect[] = await getFromCMS(redirectQuery);
|
||||
|
||||
const { shortname } = params;
|
||||
|
||||
CMSresponse.forEach((res) => {
|
||||
if (res.shortname == shortname) {
|
||||
// if match
|
||||
throw redirect(302, res.url);
|
||||
}
|
||||
});
|
||||
|
||||
return {
|
||||
shortname: shortname,
|
||||
availableShortnames: CMSresponse
|
||||
};
|
||||
};
|
||||
@@ -1,4 +1,4 @@
|
||||
import { getFromCMS } from 'utils/utils.js';
|
||||
import { getFromCMS } from '$lib/utils.js';
|
||||
|
||||
// needs to concat and format this text
|
||||
const query = `*[_type == "resources"]{
|
||||
|
||||
@@ -1,8 +0,0 @@
|
||||
export default interface CouncilMember {
|
||||
name: string;
|
||||
email: string;
|
||||
position: string;
|
||||
positionDescription: string;
|
||||
image: string; // URL
|
||||
yearProgram: string;
|
||||
}
|
||||
Reference in New Issue
Block a user