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

Implementation of a new react app for displaying opening hours from CMS with FullCalendar integration - Ddfform 414 #1023

Merged
Merged
Show file tree
Hide file tree
Changes from 24 commits
Commits
Show all changes
59 commits
Select commit Hold shift + click to select a range
f7c5ba6
Initialize a basic App for `Calendar`
kasperbirch1 Mar 12, 2024
a7146fb
Implement `FullCalendar` functionality
kasperbirch1 Mar 12, 2024
46f03a1
Add `height="100vh"` + `stickyHeaderDates`
kasperbirch1 Mar 12, 2024
4777b1e
Add `allDay` property to default event
kasperbirch1 Mar 12, 2024
d1d842e
Add logic that prevents events from spanning over multiple days
kasperbirch1 Mar 12, 2024
4ce0ef6
WIP: Implementing new Orval-generated hooks to retrieve CMS opening h…
kasperbirch1 Mar 13, 2024
8233f95
Add WireMock for `https://dpl-cms.docker` and add opening hours endpo…
kasperbirch1 Mar 13, 2024
bd0a53e
Use data from `useDplOpeningHoursListGET` in `Calendar` app
kasperbirch1 Mar 13, 2024
7349ded
Add monthly view to FullCalendar
kasperbirch1 Mar 13, 2024
e30239c
Remove `height="100vh"`
kasperbirch1 Mar 13, 2024
5b5872a
Change app name to `OpeningHoursEditor`
kasperbirch1 Mar 13, 2024
f6ed8a8
Add `eslint-disable-next-line no-alert`
kasperbirch1 Mar 13, 2024
3a0cf00
Add `height="auto"`to `FullCalendar`
kasperbirch1 Mar 13, 2024
79d6de0
Convert helper to TypeScript (.ts) file
kasperbirch1 Mar 13, 2024
bd1aaff
Add functionality to edit/remove events in `FullCalendar`
kasperbirch1 Mar 13, 2024
4220af3
Refactor handler functions into `helper.ts`
kasperbirch1 Mar 13, 2024
f5b57f0
Refactor event start/end comparison into `isSameDay` function
kasperbirch1 Mar 14, 2024
4668e00
Refactor `helper.ts` to utilize arrow functions
kasperbirch1 Mar 14, 2024
a1e902d
Remove unnecessary 'inline function' from `eventClick`
kasperbirch1 Mar 14, 2024
6616578
Refactored `handleEventRemove`
kasperbirch1 Mar 14, 2024
1320070
Refactor `OpeningHoursEditorEventContent` into component structure
kasperbirch1 Mar 14, 2024
d221388
Add `selectMirror` and Start/End Time to `OpeningHoursEditorEventCont…
kasperbirch1 Mar 14, 2024
77cbc69
Add `useOpeningHours` hook
kasperbirch1 Mar 15, 2024
25676c2
Refactor `useOpeningHours` hook to fetch opening hours data internally
kasperbirch1 Mar 15, 2024
04316ad
Refactor `isSameDay` to `adjustEndDateToStartDay`
kasperbirch1 Mar 22, 2024
d8ddd99
Move event dates to week 12 in WireMock
kasperbirch1 Mar 20, 2024
70e4c65
POC: Add `Dialog`
kasperbirch1 Mar 20, 2024
00e8bc5
Get rid of TypeScript ESLint errors.
kasperbirch1 Mar 20, 2024
84ba0fd
Add form elements for event editing in `Dialog`
kasperbirch1 Mar 21, 2024
60a7f86
Implemented `DialogFormularAdd` / `DialogFormularEdit`
kasperbirch1 Mar 25, 2024
7e09799
Update to latest openAPI spec from DPL-CMS
kasperbirch1 Mar 25, 2024
5e56748
Use date objects in helper functions
kasperbirch1 Mar 26, 2024
fd6cc81
Fix comment casing in Dialog component
kasperbirch1 Mar 26, 2024
921442c
Move `Dialog` to components folder for shared use
kasperbirch1 Mar 26, 2024
bbfe79b
Move 'Remove event' button inside `DialogFormularEdit`
kasperbirch1 Mar 26, 2024
8a3105f
Set `allDay: false` for Event addition
kasperbirch1 Mar 26, 2024
0c988ce
Disable all-day slot in the `FullCalendar`
kasperbirch1 Mar 26, 2024
ae3f072
Update to Latest OpenAPI Spec and Refactor `handleEventRemove`
kasperbirch1 Mar 26, 2024
963734c
Update wiremock for `useDplOpeningHoursListGET``
kasperbirch1 Mar 26, 2024
df3be2f
WIP: Refactor `handleEventAdd`
kasperbirch1 Mar 26, 2024
a814e73
Fix date selection Issue + Add `useEscapeKey` hook to `Dialog`
kasperbirch1 Apr 2, 2024
2e52b7c
Refactor `useOpeningHours` + Add logic for `handleEventEditing`
kasperbirch1 Mar 28, 2024
f19326b
Refactor Add/edit event logic for functionality and code clarity
kasperbirch1 Mar 31, 2024
9807ed3
Refactor `EventForm` to use CSS classes and `children` prop
kasperbirch1 Apr 2, 2024
bc52558
Rearrange code for improved readability
kasperbirch1 Apr 2, 2024
11b039e
Add `withConfig` to handle props from dpl-cms
kasperbirch1 Apr 3, 2024
309420e
Add `isSameTime` to disable button in `EventForm`
kasperbirch1 Apr 4, 2024
7a3f07c
Add CSS class and simplify`OpeningHoursEditorEventContent`
kasperbirch1 Apr 4, 2024
baccd3f
Upgrade `orval` and Execute `yarn codegen:client:dpl-cms`
kasperbirch1 Apr 4, 2024
69d878f
Rename `useOpeningHours` to `useOpeningHoursEditor`
kasperbirch1 Apr 4, 2024
687b1ef
Merge pull request #1048 from danskernesdigitalebibliotek/DDFFORM-500…
kasperbirch1 Apr 5, 2024
625b22b
Merge branch 'release/form-sprint-10' into DDFFORM-414-kalendervisnin…
kasperbirch1 Apr 5, 2024
0220c48
Fix Comments in Pull Request
kasperbirch1 Apr 5, 2024
5fc436f
Remove 'orval' from dependencies
kasperbirch1 Apr 5, 2024
f0e9bd0
Merge branch 'release/form-sprint-10' into DDFFORM-414-kalendervisnin…
kasperg Apr 7, 2024
4d9609e
Delete CMS model before generating a new one with Orval
kasperg Apr 7, 2024
640238e
Update our generated CMS client code
kasperg Apr 7, 2024
9763866
Set initial date through a prop in opening hours editor
kasperg Apr 7, 2024
a197f8c
Update opening hours deleted stub to match spec
kasperg Apr 7, 2024
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
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
{
"id" : "479e26d6-5197-4d8f-a2dd-fefe64aabc14",
"name" : "Opening hours",
"request" : {
"url" : "/dpl_opening_hours",
"method" : "GET"
},
"response" : {
"status" : 200,
"body" : "[\n {\n \"category\": {\n \"title\": \"Open\"\n },\n \"date\": \"2024-03-14\",\n \"start_time\": \"09:00\",\n \"end_time\": \"17:00\"\n },\n {\n \"category\": {\n \"title\": \"Open\"\n },\n \"date\": \"2024-03-15\",\n \"start_time\": \"09:00\",\n \"end_time\": \"17:00\"\n }\n]",
"headers" : { }
},
"uuid" : "479e26d6-5197-4d8f-a2dd-fefe64aabc14",
"persistent" : true,
"priority" : 5,
"insertionIndex" : 0,
"postServeActions" : [ ]
}
5 changes: 5 additions & 0 deletions .docker/wiremock/services.json
Original file line number Diff line number Diff line change
Expand Up @@ -9,5 +9,10 @@
"name" : "publizon",
"port" : 8110,
"rootDir" : "publizon"
}, {
"id" : "9d646",
"name" : "cms",
"port" : 8120,
"rootDir" : "cms"
} ]
}
1 change: 1 addition & 0 deletions .env.example
Original file line number Diff line number Diff line change
Expand Up @@ -2,3 +2,4 @@ STORYBOOK_CLIENT_ID=bdc1d425-9117-4867-acb9-c7307ddc2e0f
STORYBOOK_LIBRARY_TOKEN=e26e4ca8934475e6a9aytefb028b2e9fa846b5b8
PUBLIZON_BASEURL=http://publizon-mock.docker
FBS_BASEURL=http://fbs-mock.docker
CMS_BASEURL=http://cms-mock.docker
20 changes: 19 additions & 1 deletion docker-compose.yml
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
version: '3.7'
version: "3.7"

services:
wiremock:
Expand All @@ -9,6 +9,8 @@ services:
- "8100"
# Publizon
- "8110"
# CMS
- "8120"
volumes:
- wiremock_data:/home/wiremock
environment:
Expand Down Expand Up @@ -37,6 +39,17 @@ services:
VIRTUAL_HOST: publizon-mock.docker
VIRTUAL_PORT: 8080

wiremock-cms:
image: wiremock/wiremock:2.32.0
command: "--enable-stub-cors --global-response-templating" # Enable automatic sending of cross-origin (CORS) response headers.
ports:
- "8080"
volumes:
- wiremock_cms_data:/home/wiremock
environment:
VIRTUAL_HOST: cms-mock.docker
VIRTUAL_PORT: 8080

volumes:
root:
driver_opts:
Expand All @@ -58,3 +71,8 @@ volumes:
type: none
device: $PWD/.docker/wiremock/publizon
o: bind
wiremock_cms_data:
driver_opts:
type: none
device: $PWD/.docker/wiremock/cms
o: bind
2 changes: 1 addition & 1 deletion orval.config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -139,7 +139,7 @@ export default defineConfig({
},
input: {
target:
"https://raw.githubusercontent.com/danskernesdigitalebibliotek/dpl-cms/develop/openapi.json",
"https://raw.githubusercontent.com/danskernesdigitalebibliotek/dpl-cms/e1edb7e35beacb93e78eef0541c8ccebad68d6f1/openapi.json",
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This must be changed back when this pr is merged

danskernesdigitalebibliotek/dpl-cms#890

converterOptions: {
indent: 2
}
Expand Down
5 changes: 5 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -145,6 +145,11 @@
},
"dependencies": {
"@danskernesdigitalebibliotek/dpl-design-system": "^0.0.0-b807dae9a38d9b7811b36f4b3f3063eec78f4f1f",
"@fullcalendar/core": "^6.1.11",
"@fullcalendar/daygrid": "^6.1.11",
"@fullcalendar/interaction": "^6.1.11",
"@fullcalendar/react": "^6.1.11",
"@fullcalendar/timegrid": "^6.1.11",
"@reach/alert": "^0.17.0",
"@reach/dialog": "^0.18.0",
"@reduxjs/toolkit": "^1.9.7",
Expand Down
16 changes: 16 additions & 0 deletions src/apps/opening-hours-editor/OpeningHoursEditor.dev.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
import { ComponentMeta, ComponentStory } from "@storybook/react";
import React from "react";
import serviceUrlArgs from "../../core/storybook/serviceUrlArgs";
import OpeningHoursEditor from "./OpeningHoursEditor.entry";

export default {
title: "Apps / OpeningHoursEditor",
component: OpeningHoursEditor,
argTypes: {
...serviceUrlArgs
}
} as ComponentMeta<typeof OpeningHoursEditor>;

export const App: ComponentStory<typeof OpeningHoursEditor> = (args) => (
<OpeningHoursEditor {...args} />
);
13 changes: 13 additions & 0 deletions src/apps/opening-hours-editor/OpeningHoursEditor.entry.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import React from "react";
import GuardedApp from "../../components/guarded-app";
import { withText } from "../../core/utils/text";
import { withUrls } from "../../core/utils/url";
import OpeningHoursEditor from "./OpeningHoursEditor";

const CalendarEntry: React.FC = () => (
<GuardedApp app="opening-hours-editor">
<OpeningHoursEditor />
</GuardedApp>
);

export default withUrls(withText(CalendarEntry));
4 changes: 4 additions & 0 deletions src/apps/opening-hours-editor/OpeningHoursEditor.mount.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
import addMount from "../../core/addMount";
import OpeningHoursEditor from "./OpeningHoursEditor.entry";

addMount({ appName: "opening-hours-editor", app: OpeningHoursEditor });
41 changes: 41 additions & 0 deletions src/apps/opening-hours-editor/OpeningHoursEditor.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
import React from "react";
import FullCalendar from "@fullcalendar/react";
import timeGridPlugin from "@fullcalendar/timegrid";
import dayGridPlugin from "@fullcalendar/daygrid";
import interactionPlugin from "@fullcalendar/interaction";
import da from "@fullcalendar/core/locales/da";
import OpeningHoursEditorEventContent from "./OpeningHoursEditorEventContent";
import useOpeningHours from "./useOpeningHours";

const OpeningHoursEditor: React.FC = () => {
const { events, handleEventSelect, handleEventClick, handleEventRemove } =
useOpeningHours();

return (
<FullCalendar
plugins={[dayGridPlugin, timeGridPlugin, interactionPlugin]}
headerToolbar={{
left: "title",
center: "prev,next today",
right: "dayGridMonth,timeGridWeek"
}}
initialView="timeGridWeek"
locale={da}
selectable
select={handleEventSelect}
eventClick={handleEventClick}
eventContent={(eventInput) =>
OpeningHoursEditorEventContent({
kasperbirch1 marked this conversation as resolved.
Show resolved Hide resolved
eventInput,
handleEventRemove
})
}
events={events}
stickyHeaderDates
height="auto"
selectMirror
/>
);
};

export default OpeningHoursEditor;
48 changes: 48 additions & 0 deletions src/apps/opening-hours-editor/OpeningHoursEditorEventContent.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import React from "react";
import { EventInput } from "@fullcalendar/core";

type OpeningHoursEditorEventContentProps = {
eventInput: EventInput;
handleEventRemove: (event: EventInput) => void;
};

const OpeningHoursEditorEventContent: React.FC<
OpeningHoursEditorEventContentProps
> = ({ eventInput, handleEventRemove }) => {
const { event } = eventInput;
return (
<div
style={{
padding: "5px 10px",
cursor: "pointer",
width: "100%"
}}
>
<b>{event.title}</b>
<div
style={{
display: "flex",
justifyContent: "space-between"
}}
>
<span>
{event.start?.toLocaleTimeString()} -{" "}
{event.end?.toLocaleTimeString()}
</span>

<span
role="button"
tabIndex={0}
onKeyDown={(e) => {
if (e.key === "Enter") handleEventRemove(event);
}}
onClick={() => handleEventRemove(event)}
>
</span>
</div>
</div>
);
};

export default OpeningHoursEditorEventContent;
29 changes: 29 additions & 0 deletions src/apps/opening-hours-editor/helper.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
import { DplOpeningHoursListGET200Item } from "../../core/dpl-cms/model";

const formatDateTimeString = (date: string, time: string): string => {
return `${date}T${time}:00`;
Comment on lines +6 to +7
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Consider using dayjs for this instead of simple string manipulation.

If we need the data to be in a specific format e.g. ISO8601 they also have helpers for this.

};

export const createCmsEventId = (title: string, startDay: Date) => {
return `${title}-${startDay.toISOString()}`;
};

export const formatCmsEventsToFullCalendar = (
data: DplOpeningHoursListGET200Item[]
) => {
return data.map(({ category, date, start_time, end_time }) => {
const startDateTime = new Date(formatDateTimeString(date, start_time));
return {
id: createCmsEventId(category.title, startDateTime),
title: category.title,
start: formatDateTimeString(date, start_time),
end: formatDateTimeString(date, end_time),
allDay: false,
color: "blue"
};
});
};

export const isSameDay = (startDay: Date, endDay: Date) => {
return startDay.toDateString() === endDay.toDateString();
};
80 changes: 80 additions & 0 deletions src/apps/opening-hours-editor/useOpeningHours.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,80 @@
import { useState, useEffect } from "react";
import { EventInput, DateSelectArg, EventClickArg } from "@fullcalendar/core";
import {
createCmsEventId,
formatCmsEventsToFullCalendar,
isSameDay
} from "./helper";
import { useDplOpeningHoursListGET } from "../../core/dpl-cms/dpl-cms";

const useOpeningHours = () => {
const { data: openingHoursData } = useDplOpeningHoursListGET();
const [events, setEvents] = useState<EventInput[]>([]);

useEffect(() => {
if (openingHoursData) {
const formattedEvents = formatCmsEventsToFullCalendar(openingHoursData);
setEvents(formattedEvents);
}
}, [openingHoursData]);

const handleEventSelect = (selectInfo: DateSelectArg) => {
// Todo: Replace prompt with a modal
// eslint-disable-next-line no-alert
const title = prompt("Please enter a new title for your event");
const calendarApi = selectInfo.view.calendar;

if (title) {
// Checks if the selected end date is different from the start day; if so, sets the end date to be the same as the start day and the end time to 00:00:00
const startDay = new Date(selectInfo.startStr);
let endDay = new Date(selectInfo.endStr);

if (!isSameDay(startDay, endDay)) {
endDay = new Date(startDay);
// Adds one day to the end day and sets the time to 00:00:00ß
endDay.setDate(endDay.getDate() + 1);
endDay.setHours(0, 0, 0);
}

setEvents([
...events,
{
title,
start: startDay.toISOString(),
end: endDay.toISOString(),
allDay: selectInfo.allDay,
color: "green",
id: createCmsEventId(title, startDay)
}
]);
}

// clear date selection
calendarApi.unselect();
};

const handleEventClick = (clickInfo: EventClickArg) => {
// eslint-disable-next-line no-alert
const newTitle = prompt(
"Enter a new title for this event",
clickInfo.event.title
);

if (newTitle) {
clickInfo.event.setProp("title", newTitle);
}
};

const handleEventRemove = (eventToRemove: EventInput) => {
setEvents(events.filter((event) => event.id !== eventToRemove.id));
};

return {
events,
handleEventSelect,
handleEventClick,
handleEventRemove
};
};

export default useOpeningHours;
Loading
Loading