Merge remote-tracking branch 'origin/dev/guerby' into dev/cedric

This commit is contained in:
cdricms
2025-01-27 15:33:39 +01:00
8 changed files with 309 additions and 89 deletions

View File

@@ -0,0 +1,33 @@
package api
import (
"context"
"log"
"net/http"
"fr.latosa-escrima/api/core"
)
func HandleDeleteEvent(w http.ResponseWriter, r *http.Request) {
uuid := r.PathValue("event_uuid")
var event core.Event
res, err := core.DB.NewDelete().
Model(&event).
Where("id = ?", uuid).
Returning("*").
Exec(context.Background())
if err != nil {
core.JSONError{
Status: core.Error,
Message: err.Error(),
}.Respond(w, http.StatusInternalServerError)
return
}
log.Println(res)
core.JSONSuccess{
Status: core.Success,
Message: "Event deleted.",
}.Respond(w, http.StatusOK)
}

View File

@@ -1,39 +0,0 @@
package api
import (
"context"
"net/http"
core "fr.latosa-escrima/api/core"
)
func HandleGetEvent(w http.ResponseWriter, r *http.Request) {
w.Header().Set("Content-Type", "application/json")
if r.Method != http.MethodGet {
core.JSONError{
Status: core.Error,
Message: "Method not Allowed",
}.Respond(w, http.StatusMethodNotAllowed)
return
}
event_uuid := r.PathValue("uuid")
var event core.Event
_, err := core.DB.NewSelect().Model(&event).Where("uuid = ?", event_uuid).ScanAndCount(context.Background())
if err != nil {
core.JSONError{
Status: core.Error,
Message: err.Error(),
}.Respond(w, http.StatusNotAcceptable)
return
}
core.JSONSuccess{
Status: core.Success,
Message: "Event Returned",
Data: event,
}.Respond(w, http.StatusOK)
return
}

49
backend/api/get_event.go Normal file
View File

@@ -0,0 +1,49 @@
package api
import (
"context"
"fmt"
"net/http"
core "fr.latosa-escrima/api/core"
)
func HandleGetEvent(w http.ResponseWriter, r *http.Request) {
event_uuid := r.PathValue("event_uuid")
var event core.Event
_, err := core.DB.NewSelect().Model(&event).Where("uuid = ?", event_uuid).ScanAndCount(context.Background())
if err != nil {
core.JSONError{
Status: core.Error,
Message: err.Error(),
}.Respond(w, http.StatusInternalServerError)
return
}
core.JSONSuccess{
Status: core.Success,
Message: "Event successfully sent",
Data: event,
}.Respond(w, http.StatusOK)
return
}
func HangleGetEvents(w http.ResponseWriter, r *http.Request) {
var events []core.Event
rowsCount, err := core.DB.NewSelect().Model(&events).ScanAndCount(context.Background())
if err != nil {
core.JSONError{
Status: core.Error,
Message: err.Error(),
}.Respond(w, http.StatusInternalServerError)
return
}
core.JSONSuccess{
Status: core.Success,
Message: fmt.Sprintf("%d Event successfully sent", rowsCount),
Data: events,
}.Respond(w, http.StatusOK)
return
}

View File

@@ -2,36 +2,20 @@ package api
import ( import (
"context" "context"
"net/http"
"io"
"encoding/json" "encoding/json"
"net/http"
core "fr.latosa-escrima/api/core" core "fr.latosa-escrima/api/core"
) )
func HandleCreateEvent(w http.ResponseWriter, r *http.Request) { func HandleCreateEvent(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost { var event core.Event
core.JSONError{ err := json.NewDecoder(r.Body).Decode(&event)
Status: core.Error,
Message: "Method is not allowed",
}.Respond(w, http.StatusMethodNotAllowed)
return
}
body, err := io.ReadAll(r.Body)
if err != nil { if err != nil {
core.JSONError{ core.JSONError{
Status: core.Error, Status: core.Error,
Message: err.Error(), Message: err.Error(),
}.Respond(w, http.StatusNoContent) }.Respond(w, http.StatusBadRequest)
return
}
var event core.Event
if err = json.Unmarshal(body, &event); err != nil {
core.JSONError{
Status: core.Error,
Message: err.Error(),
}.Respond(w, http.StatusNoContent)
return return
} }
@@ -45,7 +29,7 @@ func HandleCreateEvent(w http.ResponseWriter, r *http.Request) {
core.JSONSuccess{ core.JSONSuccess{
Status: core.Success, Status: core.Success,
Message: "Event inserted", Message: "Event created",
Data: event, Data: event,
}.Respond(w, http.StatusCreated) }.Respond(w, http.StatusCreated)
} }

View File

@@ -0,0 +1,65 @@
package api
import (
"context"
"encoding/json"
"log"
"net/http"
"fr.latosa-escrima/api/core"
"github.com/google/uuid"
)
func HandleUpdateEvent(w http.ResponseWriter, r *http.Request) {
var event core.Event
err := json.NewDecoder(r.Body).Decode(&event)
if err != nil {
core.JSONError{
Status: core.Error,
Message: err.Error(),
}.Respond(w, http.StatusBadRequest)
return
}
event_uuid := r.PathValue("event_uuid")
event.EventID, err = uuid.FromBytes([]byte(event_uuid))
if err != nil {
core.JSONError{
Status: core.Error,
Message: err.Error(),
}.Respond(w, http.StatusBadRequest)
return
}
log.Println(event)
// val := reflect.ValueOf(event)
// typ := val.Type()
//
// for i := 0; i < val.NumField(); i++ {
// field := val.Field(i)
// fieldType := typ.Field(i)
// fmt.Printf("Field Name: %s, Field Value: %v\n", fieldType.Name, field.Interface())
// if fiel
// }
_, err = core.DB.NewUpdate().
Model(event).
OmitZero().
WherePK().
Exec(context.Background())
if err != nil {
core.JSONError{
Status: core.Error,
Message: "Event not found.",
}.Respond(w, http.StatusInternalServerError)
return
}
core.JSONSuccess{
Status: core.Success,
Message: "Event updated",
Data: event,
}.Respond(w, http.StatusOK)
}

View File

@@ -105,11 +105,24 @@ func main() {
"/users/{user_uuid}/update": { "/users/{user_uuid}/update": {
Handler: api.HandleUpdateUser, Handler: api.HandleUpdateUser,
Middlewares: []core.Middleware{api.Methods("PATCH"), api.AuthJWT}}, Middlewares: []core.Middleware{api.Methods("PATCH"), api.AuthJWT}},
// "/users/{user_uuid}/events": {Handler: nil, Middleware: nil}, "/events": {
// "/users/{user_uuid}/events/{event_uuid}": {Handler: nil, Middleware: nil}, Handler: api.HangleGetEvents,
// "/users/{user_uuid}/events/{event_uuid}/delete": {Handler: nil, Middleware: nil}, Middlewares: []core.Middleware{api.Methods("GET")}},
// "/users/{user_uuid}/events/{event_uuid}/update": {Handler: nil, Middleware: nil}, "/events/new": {
"/blogs/new": {Handler: api.HandleCreateBlog, Middlewares: nil}, Handler: api.HandleCreateEvent,
Middlewares: []core.Middleware{api.Methods("POST")}},
"/events/{event_uuid}": {
Handler: api.HandleGetEvent,
Middlewares: []core.Middleware{api.Methods("GET")}},
"/events/{event_uuid}/delete": {
Handler: api.HandleDeleteEvent,
Middlewares: []core.Middleware{api.Methods("DELETE")}},
"/events/{event_uuid}/update": {
Handler: api.HandleUpdateEvent,
Middlewares: []core.Middleware{api.Methods("PATCH")}},
"/blogs/new": {
Handler: api.HandleCreateBlog,
Middlewares: []core.Middleware{api.Methods(("POST"))}},
"/blogs/{uuid}": { "/blogs/{uuid}": {
Handler: api.HandleGetBlog, Handler: api.HandleGetBlog,
Middlewares: []core.Middleware{api.Methods("GET")}}, Middlewares: []core.Middleware{api.Methods("GET")}},

View File

@@ -1,5 +1,6 @@
"use client"; "use client";
import { request, useApi } from "@/hooks/use-api";
import "@schedule-x/theme-shadcn/dist/index.css"; import "@schedule-x/theme-shadcn/dist/index.css";
import { useNextCalendarApp, ScheduleXCalendar } from "@schedule-x/react"; import { useNextCalendarApp, ScheduleXCalendar } from "@schedule-x/react";
import { createEventsServicePlugin } from "@schedule-x/events-service"; import { createEventsServicePlugin } from "@schedule-x/events-service";
@@ -29,6 +30,7 @@ import {
import { CalendarIcon } from "lucide-react"; import { CalendarIcon } from "lucide-react";
import { Calendar } from "@/components/ui/calendar"; import { Calendar } from "@/components/ui/calendar";
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
import { requestFormReset } from "react-dom";
const Planning = () => { const Planning = () => {
const plugins = [createEventsServicePlugin()]; const plugins = [createEventsServicePlugin()];
@@ -36,7 +38,7 @@ const Planning = () => {
useState<CalendarEventExternal | null>(null); useState<CalendarEventExternal | null>(null);
const [events, setEvents] = useState<CalendarEventExternal[]>([ const [events, setEvents] = useState<CalendarEventExternal[]>([
{ {
id: "1", id: "1", // TODO put an uuid there
title: "Event 1", title: "Event 1",
start: format(new Date(Date.now()), "yyyy-MM-dd HH:mm"), start: format(new Date(Date.now()), "yyyy-MM-dd HH:mm"),
end: format( end: format(
@@ -45,6 +47,9 @@ const Planning = () => {
), ),
}, },
]); ]);
const [requestCreateEvent, setRequestCreateEvent] = useState(false)
const calendar = useNextCalendarApp( const calendar = useNextCalendarApp(
{ {
theme: "shadcn", theme: "shadcn",
@@ -62,11 +67,47 @@ const Planning = () => {
onEventClick(event, e) { onEventClick(event, e) {
setEventSelected(event); setEventSelected(event);
}, },
async onClickDateTime(dateTime) {
setRequestCreateEvent(true)
const newEvent: CalendarEventExternal = {
id: "5",
title: "Event 1",
start: dateTime,
end: format(
new Date(dateTime).getTime() + (1000 * 60 * 60),
"yyyy-MM-dd HH:mm",
),
}
try {
const res = await request<undefined>(
`/events/new`,
{
method: "POST",
body: JSON.stringify(newEvent),
requiresAuth: true,
csrfToken: false
},)
if (res.status === "Error") {
console.log("Error")
}
if (res.status === "Success") {
console.log("Success")
}
} catch (e) {
console.log(e)
}
},
}, },
}, },
plugins, plugins,
); );
const {data: requestedEvents, isLoading, success} = useApi("/events", {
onSuccess: (data) => {
calendar?.events.set(data)
}
}, false, false)
useEffect(() => { useEffect(() => {
// get all events // get all events
calendar?.events.getAll(); calendar?.events.getAll();
@@ -78,7 +119,7 @@ const Planning = () => {
<ScheduleXCalendar calendarApp={calendar} /> <ScheduleXCalendar calendarApp={calendar} />
</div> </div>
<Dialog <Dialog
open={eventSelected !== null} open={eventSelected !== null || false}
onOpenChange={(open) => { onOpenChange={(open) => {
setEventSelected((e) => (open ? e : null)); setEventSelected((e) => (open ? e : null));
}} }}
@@ -137,10 +178,9 @@ const Planning = () => {
</Popover> */} </Popover> */}
<Input <Input
id="start" id="start"
value={eventSelected?.start} value={eventSelected?.start || ""}
onChange={(e) => { onChange={(e) => {
const val = e.currentTarget.value; const val = e.currentTarget.value;
console.log(val);
setEventSelected((ev) => { setEventSelected((ev) => {
if (ev) if (ev)
return { return {
@@ -159,35 +199,72 @@ const Planning = () => {
</Label> </Label>
<Input <Input
id="end" id="end"
value={eventSelected?.end} value={eventSelected?.end || ""}
onChange={(e) => onChange={(e) => {
const val = e.currentTarget.value
setEventSelected((ev) => { setEventSelected((ev) => {
if (ev) if (ev)
return { return {
...ev, ...ev,
end: e.currentTarget.value, end: val,
}; };
return ev; return ev;
}) })
} }}
className="col-span-3" className="col-span-3"
/> />
</div> </div>
</div> </div>
<DialogFooter> <DialogFooter className="flex flex-row justify-end">
<Button <Button className="bg-red-700"
onClick={() => { onClick={async () => {
setEvents((evs) => { calendar?.events?.remove(eventSelected!.id)
evs = evs.filter( try {
(e) => e.id !== eventSelected?.id, const res = await request<undefined>(
); `/events/${eventSelected!.id}/delete`,
evs.push(eventSelected!); {
return evs; method: "DELETE",
}); body: JSON.stringify(eventSelected),
requiresAuth: false,
csrfToken: false
},)
if (res.status === "Error") {
console.log("Error")
}
if (res.status === "Success") {
console.log("Success")
}
} catch (e) {
console.log(e)
}
}} }}
type="submit" type="submit">
> Supprimer
Mettre à jour </Button>
<Button className="bg-blue-500"
onClick={async () => {
calendar?.events?.update(eventSelected!)
try {
const res = await request<undefined>(
`/events/${eventSelected!.id}/update`,
{
method: "PATCH",
body: JSON.stringify(eventSelected),
requiresAuth: true,
csrfToken: false
},)
if (res.status === "Error") {
console.log("Error")
}
if (res.status === "Success") {
console.log("Success")
}
} catch (e) {
console.log(e)
}
}}
type="submit">
Actualiser
</Button> </Button>
</DialogFooter> </DialogFooter>
</DialogContent> </DialogContent>

38
frontend/hooks/events.tsx Normal file
View File

@@ -0,0 +1,38 @@
"use client";
import { setCookie } from "cookies-next";
import { useEffect, useState } from "react";
import { API_URL } from "@/lib/constants";
export interface LoginArgs {
email: string;
password: string;
}
export default function useLogin() {
const {
data,
isLoading: loading,
isSuccess,
} = request<string, LoginArgs>(
"/users/login",
undefined,
"POST",
false,
true,
);
const login = async (inputs: LoginArgs) => {
try {
const res = await trigger(inputs);
if (!res) throw new Error("The server hasn't responded.");
if (res.status === "Error") throw new Error(res.message);
if (res.data) setCookie("auth_token", res.data);
return res;
} catch (error: any) {
throw new Error(error.message);
}
};
return { login, loading, isSuccess };
}