feat: add guest comb to competition register form

This commit is contained in:
Thibaut Valentin 2025-11-30 18:38:18 +01:00
parent f6d4bb0fe4
commit 936392f8bd
7 changed files with 286 additions and 64 deletions

View File

@ -1,9 +1,6 @@
package fr.titionfire.ffsaf.domain.service; package fr.titionfire.ffsaf.domain.service;
import fr.titionfire.ffsaf.data.model.CompetitionModel; import fr.titionfire.ffsaf.data.model.*;
import fr.titionfire.ffsaf.data.model.HelloAssoRegisterModel;
import fr.titionfire.ffsaf.data.model.MembreModel;
import fr.titionfire.ffsaf.data.model.RegisterModel;
import fr.titionfire.ffsaf.data.repository.*; import fr.titionfire.ffsaf.data.repository.*;
import fr.titionfire.ffsaf.net2.ServerCustom; import fr.titionfire.ffsaf.net2.ServerCustom;
import fr.titionfire.ffsaf.net2.data.SimpleCompet; import fr.titionfire.ffsaf.net2.data.SimpleCompet;
@ -16,10 +13,9 @@ import fr.titionfire.ffsaf.rest.data.SimpleCompetData;
import fr.titionfire.ffsaf.rest.data.SimpleRegisterComb; import fr.titionfire.ffsaf.rest.data.SimpleRegisterComb;
import fr.titionfire.ffsaf.rest.exception.DBadRequestException; import fr.titionfire.ffsaf.rest.exception.DBadRequestException;
import fr.titionfire.ffsaf.rest.exception.DForbiddenException; import fr.titionfire.ffsaf.rest.exception.DForbiddenException;
import fr.titionfire.ffsaf.utils.CompetitionSystem; import fr.titionfire.ffsaf.rest.exception.DNotFoundException;
import fr.titionfire.ffsaf.utils.RegisterMode; import fr.titionfire.ffsaf.utils.*;
import fr.titionfire.ffsaf.utils.SecurityCtx; import fr.titionfire.ffsaf.ws.send.SRegister;
import fr.titionfire.ffsaf.utils.Utils;
import io.quarkus.cache.Cache; import io.quarkus.cache.Cache;
import io.quarkus.cache.CacheName; import io.quarkus.cache.CacheName;
import io.quarkus.hibernate.reactive.panache.Panache; import io.quarkus.hibernate.reactive.panache.Panache;
@ -64,6 +60,9 @@ public class CompetitionService {
@Inject @Inject
CombRepository combRepository; CombRepository combRepository;
@Inject
CompetitionGuestRepository competitionGuestRepository;
@Inject @Inject
ServerCustom serverCustom; ServerCustom serverCustom;
@ -83,6 +82,9 @@ public class CompetitionService {
@Inject @Inject
Vertx vertx; Vertx vertx;
@Inject
SRegister sRegister;
@Inject @Inject
@CacheName("safca-config") @CacheName("safca-config")
Cache cache; Cache cache;
@ -108,7 +110,8 @@ public class CompetitionService {
} }
return permService.hasAdminViewPerm(securityCtx, id) return permService.hasAdminViewPerm(securityCtx, id)
.chain(competitionModel -> Mutiny.fetch(competitionModel.getInsc()) .chain(competitionModel -> Mutiny.fetch(competitionModel.getInsc())
.map(insc -> CompetitionData.fromModel(competitionModel).addInsc(insc))) .chain(insc -> Mutiny.fetch(competitionModel.getGuests())
.map(guest -> CompetitionData.fromModel(competitionModel).addInsc(insc, guest))))
.chain(data -> .chain(data ->
vertx.getOrCreateContext().executeBlocking(() -> { vertx.getOrCreateContext().executeBlocking(() -> {
keycloakService.getUser(UUID.fromString(data.getOwner())) keycloakService.getUser(UUID.fromString(data.getOwner()))
@ -175,6 +178,7 @@ public class CompetitionService {
model.setSystem(data.getSystem()); model.setSystem(data.getSystem());
model.setClub(clubModel); model.setClub(clubModel);
model.setInsc(new ArrayList<>()); model.setInsc(new ArrayList<>());
model.setGuests(new ArrayList<>());
model.setUuid(UUID.randomUUID().toString()); model.setUuid(UUID.randomUUID().toString());
model.setOwner(securityCtx.getSubject()); model.setOwner(securityCtx.getSubject());
@ -235,11 +239,18 @@ public class CompetitionService {
public Uni<List<SimpleRegisterComb>> getRegister(SecurityCtx securityCtx, Long id, String source) { public Uni<List<SimpleRegisterComb>> getRegister(SecurityCtx securityCtx, Long id, String source) {
if ("admin".equals(source)) if ("admin".equals(source))
return permService.hasEditPerm(securityCtx, id) return permService.hasEditPerm(securityCtx, id)
.chain(c -> Mutiny.fetch(c.getInsc())) .chain(c -> {
.onItem().transformToMulti(Multi.createFrom()::iterable) Uni<List<SimpleRegisterComb>> uni = Mutiny.fetch(c.getInsc())
.onItem().call(combModel -> Mutiny.fetch(combModel.getMembre().getLicences())) .onItem().transformToMulti(Multi.createFrom()::iterable)
.map(combModel -> SimpleRegisterComb.fromModel(combModel, combModel.getMembre().getLicences())) .onItem().call(combModel -> Mutiny.fetch(combModel.getMembre().getLicences()))
.collect().asList(); .map(cm -> SimpleRegisterComb.fromModel(cm, cm.getMembre().getLicences()))
.collect().asList();
return uni
.call(l -> Mutiny.fetch(c.getGuests())
.map(guest -> guest.stream().map(SimpleRegisterComb::fromModel).toList())
.invoke(l::addAll));
});
if ("club".equals(source)) if ("club".equals(source))
return Uni.createFrom().nullItem() return Uni.createFrom().nullItem()
.invoke(Unchecked.consumer(__ -> { .invoke(Unchecked.consumer(__ -> {
@ -262,17 +273,44 @@ public class CompetitionService {
public Uni<SimpleRegisterComb> addRegisterComb(SecurityCtx securityCtx, Long id, RegisterRequestData data, public Uni<SimpleRegisterComb> addRegisterComb(SecurityCtx securityCtx, Long id, RegisterRequestData data,
String source) { String source) {
if ("admin".equals(source)) if ("admin".equals(source))
return permService.hasEditPerm(securityCtx, id) if (data.getLicence() != -1) { // not a guest
.chain(c -> findComb(data.getLicence(), data.getFname(), data.getLname()) return permService.hasEditPerm(securityCtx, id)
.call(combModel -> { .chain(c -> findComb(data.getLicence(), data.getFname(), data.getLname())
if (c.getBanMembre() == null) .call(combModel -> Mutiny.fetch(combModel.getLicences()))
c.setBanMembre(new ArrayList<>()); .call(combModel -> {
c.getBanMembre().remove(combModel.getId()); if (c.getBanMembre() == null)
return Panache.withTransaction(() -> repository.persist(c)); c.setBanMembre(new ArrayList<>());
}) c.getBanMembre().remove(combModel.getId());
.chain(combModel -> updateRegister(data, c, combModel, true))) return Panache.withTransaction(() -> repository.persist(c));
.chain(r -> Mutiny.fetch(r.getMembre().getLicences()) })
.map(licences -> SimpleRegisterComb.fromModel(r, licences))); .chain(combModel -> updateRegister(data, c, combModel, true)))
.map(r -> SimpleRegisterComb.fromModel(r, r.getMembre().getLicences()));
} else {
return permService.hasEditPerm(securityCtx, id)
.chain(c -> competitionGuestRepository.findById(data.getId() * -1)
.map(g -> {
if (g != null)
return g;
CompetitionGuestModel model = new CompetitionGuestModel();
model.setCompetition(c);
return model;
}))
.chain(model -> {
model.setFname(data.getFname());
model.setLname(data.getLname());
model.setGenre(data.getGenre());
model.setClub(data.getClub());
model.setCountry(data.getCountry());
model.setWeight(data.getWeight());
model.setCategorie(data.getCategorie());
return Panache.withTransaction(() -> competitionGuestRepository.persist(model))
.call(r -> model.getCompetition().getSystem() == CompetitionSystem.INTERNAL ?
sRegister.sendRegister(model.getCompetition().getUuid(),
r) : Uni.createFrom().voidItem());
})
.map(SimpleRegisterComb::fromModel);
}
if ("club".equals(source)) if ("club".equals(source))
return repository.findById(id) return repository.findById(id)
.invoke(Unchecked.consumer(cm -> { .invoke(Unchecked.consumer(cm -> {
@ -283,6 +321,7 @@ public class CompetitionService {
throw new DBadRequestException("Inscription fermée"); throw new DBadRequestException("Inscription fermée");
})) }))
.chain(c -> findComb(data.getLicence(), data.getFname(), data.getLname()) .chain(c -> findComb(data.getLicence(), data.getFname(), data.getLname())
.call(combModel -> Mutiny.fetch(combModel.getLicences()))
.invoke(Unchecked.consumer(model -> { .invoke(Unchecked.consumer(model -> {
if (!securityCtx.isInClubGroup(model.getClub().getId())) if (!securityCtx.isInClubGroup(model.getClub().getId()))
throw new DForbiddenException(); throw new DForbiddenException();
@ -291,8 +330,7 @@ public class CompetitionService {
"Vous n'avez pas le droit d'inscrire ce membre (par décision de l'administrateur de la compétition)"); "Vous n'avez pas le droit d'inscrire ce membre (par décision de l'administrateur de la compétition)");
})) }))
.chain(combModel -> updateRegister(data, c, combModel, false))) .chain(combModel -> updateRegister(data, c, combModel, false)))
.chain(r -> Mutiny.fetch(r.getMembre().getLicences()) .map(r -> SimpleRegisterComb.fromModel(r, r.getMembre().getLicences()));
.map(licences -> SimpleRegisterComb.fromModel(r, licences)));
return repository.findById(id) return repository.findById(id)
.invoke(Unchecked.consumer(cm -> { .invoke(Unchecked.consumer(cm -> {
@ -347,15 +385,18 @@ public class CompetitionService {
SReqRegister.sendIfNeed(serverCustom.clients, SReqRegister.sendIfNeed(serverCustom.clients,
new CompetitionData.SimpleRegister(r.getMembre().getId(), new CompetitionData.SimpleRegister(r.getMembre().getId(),
r.getOverCategory(), r.getWeight(), r.getCategorie(), r.getOverCategory(), r.getWeight(), r.getCategorie(),
(r.getClub() == null) ? null : r.getClub().getId()), c.getId()); (r.getClub() == null) ? null : r.getClub().getId(),
(r.getClub() == null) ? null : r.getClub().getName()), c.getId());
} }
return r; return r;
})) }))
.chain(r -> Panache.withTransaction(() -> registerRepository.persist(r))); .chain(r -> Panache.withTransaction(() -> registerRepository.persist(r)))
.call(r -> c.getSystem() == CompetitionSystem.INTERNAL ?
sRegister.sendRegister(c.getUuid(), r) : Uni.createFrom().voidItem());
} }
private Uni<MembreModel> findComb(Long licence, String fname, String lname) { private Uni<MembreModel> findComb(Long licence, String fname, String lname) {
if (licence != null && licence != 0) { if (licence != null && licence > 0) {
return combRepository.find("licence = ?1", licence).firstResult() return combRepository.find("licence = ?1", licence).firstResult()
.invoke(Unchecked.consumer(combModel -> { .invoke(Unchecked.consumer(combModel -> {
if (combModel == null) if (combModel == null)
@ -398,30 +439,45 @@ public class CompetitionService {
if (new Date().before(cm.getStartRegister()) || new Date().after(cm.getEndRegister())) if (new Date().before(cm.getStartRegister()) || new Date().after(cm.getEndRegister()))
throw new DBadRequestException("Inscription fermée"); throw new DBadRequestException("Inscription fermée");
})) }))
.call(cm -> membreService.getByAccountId(securityCtx.getSubject()) .call(cm -> membreService.getById(combId)
.invoke(Unchecked.consumer(model -> { .invoke(Unchecked.consumer(model -> {
if (model == null)
throw new DNotFoundException("Membre " + combId + " n'existe pas");
if (!securityCtx.isInClubGroup(model.getClub().getId())) if (!securityCtx.isInClubGroup(model.getClub().getId()))
throw new DForbiddenException(); throw new DForbiddenException();
}))) })))
.chain(c -> deleteRegister(combId, c, false)); .chain(c -> deleteRegister(combId, c, false));
return repository.findById(id) return repository.findById(id)
.invoke(Unchecked.consumer(cm -> { .call(cm -> membreService.getByAccountId(securityCtx.getSubject())
if (cm.getRegisterMode() != RegisterMode.FREE) .invoke(Unchecked.consumer(model -> {
throw new DForbiddenException(); if (cm.getRegisterMode() != RegisterMode.FREE || !Objects.equals(model.getId(), combId))
if (new Date().before(cm.getStartRegister()) || new Date().after(cm.getEndRegister())) throw new DForbiddenException();
throw new DBadRequestException("Inscription fermée"); if (new Date().before(cm.getStartRegister()) || new Date().after(cm.getEndRegister()))
})) throw new DBadRequestException("Inscription fermée");
})))
.chain(c -> deleteRegister(combId, c, false)); .chain(c -> deleteRegister(combId, c, false));
} }
private Uni<Void> deleteRegister(Long combId, CompetitionModel c, boolean admin) { private Uni<Void> deleteRegister(Long combId, CompetitionModel c, boolean admin) {
if (admin && combId < 0) {
return competitionGuestRepository.find("competition = ?1 AND id = ?2", c, combId * -1).firstResult()
.onFailure().transform(t -> new DBadRequestException("Combattant non inscrit"))
.call(Unchecked.function(
model -> Panache.withTransaction(() -> competitionGuestRepository.delete(model))
.call(r -> c.getSystem() == CompetitionSystem.INTERNAL ?
sRegister.sendRegisterRemove(c.getUuid(), combId) : Uni.createFrom()
.voidItem())))
.replaceWithVoid();
}
return registerRepository.find("competition = ?1 AND membre.id = ?2", c, combId).firstResult() return registerRepository.find("competition = ?1 AND membre.id = ?2", c, combId).firstResult()
.onFailure().transform(t -> new DBadRequestException("Combattant non inscrit")) .onFailure().transform(t -> new DBadRequestException("Combattant non inscrit"))
.call(Unchecked.function(registerModel -> { .call(Unchecked.function(registerModel -> {
if (!admin && registerModel.isLockEdit()) if (!admin && registerModel.isLockEdit())
throw new DForbiddenException("Modification bloquée par l'administrateur de la compétition"); throw new DForbiddenException("Modification bloquée par l'administrateur de la compétition");
return Panache.withTransaction(() -> registerRepository.delete(registerModel)); return Panache.withTransaction(() -> registerRepository.delete(registerModel))
.call(r -> c.getSystem() == CompetitionSystem.INTERNAL ?
sRegister.sendRegisterRemove(c.getUuid(), combId) : Uni.createFrom().voidItem());
})) }))
.replaceWithVoid(); .replaceWithVoid();
} }
@ -528,8 +584,13 @@ public class CompetitionService {
continue; continue;
uni = uni.call(__ -> Panache.withTransaction( uni = uni.call(__ -> Panache.withTransaction(
() -> registerRepository.delete("competition = ?1 AND membre = ?2", () -> registerRepository.delete("competition = ?1 AND membre = ?2",
reg.getCompetition(), reg.getMembre()))); reg.getCompetition(), reg.getMembre())))
.call(r -> reg.getCompetition().getSystem() == CompetitionSystem.INTERNAL ?
sRegister.sendRegisterRemove(reg.getCompetition().getUuid(),
reg.getMembre().getId()) : Uni.createFrom().voidItem()).onFailure()
.recoverWithNull()
;
} }
return uni; return uni;
@ -541,7 +602,8 @@ public class CompetitionService {
public Uni<Response> registerHelloAsso(NotificationData data) { public Uni<Response> registerHelloAsso(NotificationData data) {
String organizationSlug = data.getOrganizationSlug(); String organizationSlug = data.getOrganizationSlug();
String formSlug = data.getFormSlug(); String formSlug = data.getFormSlug();
RegisterRequestData req = new RegisterRequestData(null, "", "", null, 0, false); RegisterRequestData req = new RegisterRequestData(null, "", "", null, 0, false, null, Categorie.CADET, Genre.NA,
null, "fr");
return repository.find("data1 = ?1 AND data2 = ?2", organizationSlug, formSlug).firstResult() return repository.find("data1 = ?1 AND data2 = ?2", organizationSlug, formSlug).firstResult()
.onFailure().recoverWithNull() .onFailure().recoverWithNull()

View File

@ -1,5 +1,6 @@
package fr.titionfire.ffsaf.rest.data; package fr.titionfire.ffsaf.rest.data;
import fr.titionfire.ffsaf.data.model.CompetitionGuestModel;
import fr.titionfire.ffsaf.data.model.CompetitionModel; import fr.titionfire.ffsaf.data.model.CompetitionModel;
import fr.titionfire.ffsaf.data.model.RegisterModel; import fr.titionfire.ffsaf.data.model.RegisterModel;
import fr.titionfire.ffsaf.utils.Categorie; import fr.titionfire.ffsaf.utils.Categorie;
@ -11,6 +12,7 @@ import lombok.Data;
import java.util.Date; import java.util.Date;
import java.util.List; import java.util.List;
import java.util.stream.Stream;
@Data @Data
@AllArgsConstructor @AllArgsConstructor
@ -57,9 +59,9 @@ public class CompetitionData {
model.getAdresse(), "", model.getDate(), model.getTodate(), null, model.getAdresse(), "", model.getDate(), model.getTodate(), null,
model.getRegisterMode(), model.getStartRegister(), model.getEndRegister(), model.isPublicVisible(), model.getRegisterMode(), model.getStartRegister(), model.getEndRegister(), model.isPublicVisible(),
null, model.getClub().getName(), "", null, false, null, model.getClub().getName(), "", null, false,
"","", "",""); "", "", "", "");
if (model.getRegisterMode() == RegisterMode.HELLOASSO){ if (model.getRegisterMode() == RegisterMode.HELLOASSO) {
out.setData1(model.getData1()); out.setData1(model.getData1());
out.setData2(model.getData2()); out.setData2(model.getData2());
} }
@ -67,10 +69,15 @@ public class CompetitionData {
return out; return out;
} }
public CompetitionData addInsc(List<RegisterModel> insc) { public CompetitionData addInsc(List<RegisterModel> insc, List<CompetitionGuestModel> guests) {
this.registers = insc.stream() this.registers = Stream.concat(
.map(i -> new SimpleRegister(i.getMembre().getId(), i.getOverCategory(), i.getWeight(), insc.stream()
i.getCategorie(), (i.getClub() == null) ? null : i.getClub().getId())).toList(); .map(i -> new SimpleRegister(i.getMembre().getId(), i.getOverCategory(), i.getWeight(),
i.getCategorie(), (i.getClub() == null) ? null : i.getClub().getId(),
(i.getClub() == null) ? null : i.getClub().getName())),
guests.stream()
.map(i -> new SimpleRegister(i.getId() * -1, 0, i.getWeight(),
i.getCategorie(), null, i.getClub()))).toList();
return this; return this;
} }
@ -83,5 +90,6 @@ public class CompetitionData {
Integer weight; Integer weight;
Categorie categorie; Categorie categorie;
Long club; Long club;
String club_str;
} }
} }

View File

@ -1,5 +1,7 @@
package fr.titionfire.ffsaf.rest.data; package fr.titionfire.ffsaf.rest.data;
import fr.titionfire.ffsaf.utils.Categorie;
import fr.titionfire.ffsaf.utils.Genre;
import io.quarkus.runtime.annotations.RegisterForReflection; import io.quarkus.runtime.annotations.RegisterForReflection;
import lombok.AllArgsConstructor; import lombok.AllArgsConstructor;
import lombok.Data; import lombok.Data;
@ -17,4 +19,11 @@ public class RegisterRequestData {
private Integer weight; private Integer weight;
private int overCategory; private int overCategory;
private boolean lockEdit = false; private boolean lockEdit = false;
// for guest registration only
private Long id = null;
private Categorie categorie = Categorie.CADET;
private Genre genre = Genre.NA;
private String club = null;
private String country = null;
} }

View File

@ -1,9 +1,12 @@
package fr.titionfire.ffsaf.rest.data; package fr.titionfire.ffsaf.rest.data;
import fr.titionfire.ffsaf.data.model.CompetitionGuestModel;
import fr.titionfire.ffsaf.data.model.LicenceModel; import fr.titionfire.ffsaf.data.model.LicenceModel;
import fr.titionfire.ffsaf.data.model.MembreModel; import fr.titionfire.ffsaf.data.model.MembreModel;
import fr.titionfire.ffsaf.data.model.RegisterModel; import fr.titionfire.ffsaf.data.model.RegisterModel;
import fr.titionfire.ffsaf.net2.data.SimpleClubModel; import fr.titionfire.ffsaf.net2.data.SimpleClubModel;
import fr.titionfire.ffsaf.utils.Categorie;
import fr.titionfire.ffsaf.utils.Genre;
import fr.titionfire.ffsaf.utils.Utils; import fr.titionfire.ffsaf.utils.Utils;
import io.quarkus.runtime.annotations.RegisterForReflection; import io.quarkus.runtime.annotations.RegisterForReflection;
import lombok.AllArgsConstructor; import lombok.AllArgsConstructor;
@ -18,8 +21,9 @@ public class SimpleRegisterComb {
private long id; private long id;
private String fname; private String fname;
private String lname; private String lname;
private String genre; private Genre genre;
private String categorie; private String country;
private Categorie categorie;
private SimpleClubModel club; private SimpleClubModel club;
private Integer licence; private Integer licence;
private Integer weight; private Integer weight;
@ -30,11 +34,18 @@ public class SimpleRegisterComb {
public static SimpleRegisterComb fromModel(RegisterModel register, List<LicenceModel> licences) { public static SimpleRegisterComb fromModel(RegisterModel register, List<LicenceModel> licences) {
MembreModel membreModel = register.getMembre(); MembreModel membreModel = register.getMembre();
return new SimpleRegisterComb(membreModel.getId(), membreModel.getFname(), membreModel.getLname(), return new SimpleRegisterComb(membreModel.getId(), membreModel.getFname(), membreModel.getLname(),
membreModel.getGenre().name(), membreModel.getGenre(), membreModel.getCountry(),
(register.getCategorie() == null) ? "Catégorie inconnue" : register.getCategorie().getName(), (register.getCategorie() == null) ? null : register.getCategorie(),
SimpleClubModel.fromModel(register.getClub()), membreModel.getLicence(), register.getWeight(), SimpleClubModel.fromModel(register.getClub()), membreModel.getLicence(), register.getWeight(),
register.getOverCategory(), register.getOverCategory(),
licences.stream().anyMatch(l -> l.isValidate() && l.getSaison() == Utils.getSaison()), licences.stream().anyMatch(l -> l.isValidate() && l.getSaison() == Utils.getSaison()),
register.isLockEdit()); register.isLockEdit());
} }
public static SimpleRegisterComb fromModel(CompetitionGuestModel guest) {
return new SimpleRegisterComb(guest.getId() * -1, guest.getFname(), guest.getLname(),
guest.getGenre(), guest.getCountry(), guest.getCategorie(),
new SimpleClubModel(null, guest.getClub(), "fr", null),
null, guest.getWeight(), 0, false, false);
}
} }

View File

@ -0,0 +1,53 @@
package fr.titionfire.ffsaf.ws.send;
import fr.titionfire.ffsaf.data.model.CompetitionGuestModel;
import fr.titionfire.ffsaf.data.model.RegisterModel;
import fr.titionfire.ffsaf.domain.entity.CombEntity;
import fr.titionfire.ffsaf.net2.MessageType;
import fr.titionfire.ffsaf.ws.CompetitionWS;
import fr.titionfire.ffsaf.ws.MessageOut;
import fr.titionfire.ffsaf.ws.PermLevel;
import io.quarkus.runtime.annotations.RegisterForReflection;
import io.quarkus.websockets.next.OpenConnections;
import io.quarkus.websockets.next.UserData;
import io.smallrye.mutiny.Uni;
import jakarta.enterprise.context.ApplicationScoped;
import jakarta.inject.Inject;
import java.util.List;
import java.util.UUID;
@ApplicationScoped
@RegisterForReflection
public class SRegister {
@SuppressWarnings("CdiInjectionPointsInspection")
@Inject
OpenConnections connections;
public Uni<Void> sendRegister(String uuid, RegisterModel registerModel) {
return send(uuid, "sendRegister", CombEntity.fromModel(registerModel));
}
public Uni<Void> sendRegister(String uuid, CompetitionGuestModel model) {
return send(uuid, "sendRegister", CombEntity.fromModel(model));
}
public Uni<Void> sendRegisterRemove(String uuid, Long combId) {
return send(uuid, "sendRegister", combId);
}
public Uni<Void> send(String uuid, String code, Object data) {
List<Uni<Void>> queue = connections.findByEndpointId(CompetitionWS.class.getCanonicalName()).stream()
.filter(c -> c.pathParam("uuid").equals(uuid) && PermLevel.valueOf(
c.userData().get(UserData.TypedKey.forString("prem"))).ordinal() >= PermLevel.ADMIN.ordinal())
.map(c -> c.sendText(
new MessageOut(UUID.randomUUID(), code, MessageType.NOTIFY, data)))
.toList();
if (queue.isEmpty())
return Uni.createFrom().voidItem();
return Uni.join().all(queue).andCollectFailures().replaceWithVoid();
}
}

View File

@ -4,14 +4,14 @@ import {useFetch} from "../../hooks/useFetch.js";
import {AxiosError} from "../../components/AxiosError.jsx"; import {AxiosError} from "../../components/AxiosError.jsx";
import {ThreeDots} from "react-loader-spinner"; import {ThreeDots} from "react-loader-spinner";
import {useEffect, useReducer, useRef, useState} from "react"; import {useEffect, useReducer, useRef, useState} from "react";
import {apiAxios, errFormater} from "../../utils/Tools.js"; import {apiAxios, CatList, getCatName} from "../../utils/Tools.js";
import {toast} from "react-toastify"; import {toast} from "react-toastify";
import {SimpleReducer} from "../../utils/SimpleReducer.jsx"; import {SimpleReducer} from "../../utils/SimpleReducer.jsx";
import {FontAwesomeIcon} from "@fortawesome/react-fontawesome"; import {FontAwesomeIcon} from "@fortawesome/react-fontawesome";
import {faAdd, faGavel, faTrashCan} from "@fortawesome/free-solid-svg-icons"; import {faAdd, faGavel, faTrashCan} from "@fortawesome/free-solid-svg-icons";
import "./CompetitionRegisterAdmin.css" import "./CompetitionRegisterAdmin.css"
import * as XLSX from "xlsx-js-style"; import * as XLSX from "xlsx-js-style";
import * as Tools from "../../utils/Tools.js"; import {useCountries} from "../../hooks/useCountries.jsx";
export function CompetitionRegisterAdmin({source}) { export function CompetitionRegisterAdmin({source}) {
const {id} = useParams() const {id} = useParams()
@ -70,9 +70,14 @@ export function CompetitionRegisterAdmin({source}) {
</div> : error ? <AxiosError error={error}/> : <Def/>} </div> : error ? <AxiosError error={error}/> : <Def/>}
</div> </div>
<div className="col-lg-3"> <div className="col-lg-3">
<div className="mb-1">
<button type="button" className="btn btn-primary" data-bs-toggle="modal" data-bs-target="#registerModal"
onClick={() => setModalState({id: 0})}>Ajouter un combattant
</button>
</div>
<div className="mb-4"> <div className="mb-4">
<button type="button" className="btn btn-primary" data-bs-toggle="modal" data-bs-target="#registerModal" <button type="button" className="btn btn-primary" data-bs-toggle="modal" data-bs-target="#registerModal"
onClick={() => setModalState({})}>Ajouter un combattant onClick={() => setModalState({id: -793548328091516928})}>Ajouter un invité
</button> </button>
</div> </div>
<QuickAdd sendRegister={sendRegister} source={source}/> <QuickAdd sendRegister={sendRegister} source={source}/>
@ -276,15 +281,22 @@ const AutoCompleteInput = ({suggestions = [], handleAdd}) => {
}; };
function Modal({sendRegister, modalState, setModalState, source}) { function Modal({sendRegister, modalState, setModalState, source}) {
const country = useCountries('fr')
const [licence, setLicence] = useState("") const [licence, setLicence] = useState("")
const [fname, setFname] = useState("") const [fname, setFname] = useState("")
const [lname, setLname] = useState("") const [lname, setLname] = useState("")
const [weight, setWeight] = useState("") const [weight, setWeight] = useState("")
const [cat, setCat] = useState(0) const [cat, setCat] = useState(0)
const [gcat, setGCat] = useState("")
const [club, setClub] = useState("")
const [country_, setCountry_] = useState("fr")
const [genre, setGenre] = useState("NA")
const [editMode, setEditMode] = useState(false) const [editMode, setEditMode] = useState(false)
const [lockEdit, setLockEdit] = useState(false) const [lockEdit, setLockEdit] = useState(false)
useEffect(() => { useEffect(() => {
console.log(modalState)
if (!modalState) { if (!modalState) {
setLicence("") setLicence("")
setFname("") setFname("")
@ -293,6 +305,10 @@ function Modal({sendRegister, modalState, setModalState, source}) {
setCat(0) setCat(0)
setEditMode(false) setEditMode(false)
setLockEdit(false) setLockEdit(false)
setClub("")
setGCat("")
setCountry_("fr")
setGenre("NA")
} else { } else {
setLicence(modalState.licence ? modalState.licence : "") setLicence(modalState.licence ? modalState.licence : "")
setFname(modalState.fname ? modalState.fname : "") setFname(modalState.fname ? modalState.fname : "")
@ -301,6 +317,10 @@ function Modal({sendRegister, modalState, setModalState, source}) {
setCat(modalState.overCategory ? modalState.overCategory : 0) setCat(modalState.overCategory ? modalState.overCategory : 0)
setEditMode(modalState.licence || (modalState.fname && modalState.lname)) setEditMode(modalState.licence || (modalState.fname && modalState.lname))
setLockEdit(modalState.lockEdit) setLockEdit(modalState.lockEdit)
setClub(modalState.club ? modalState.club.name : "")
setGCat(modalState.categorie ? modalState.categorie : "")
setCountry_(modalState.country ? modalState.country : "fr")
setGenre(modalState.genre ? modalState.genre : "NA")
} }
}, [modalState]); }, [modalState]);
@ -311,46 +331,103 @@ function Modal({sendRegister, modalState, setModalState, source}) {
<form onSubmit={e => { <form onSubmit={e => {
e.preventDefault() e.preventDefault()
const new_state = { const new_state = {
licence: licence, fname: fname, lname: lname, weight: weight, overCategory: cat, lockEdit: lockEdit, id: modalState.id licence: licence,
fname: fname,
lname: lname,
weight: weight,
overCategory: cat,
lockEdit: lockEdit,
id: modalState.id !== 0 ? modalState.id : null
}
if (modalState.id < 0) {
new_state.licence = -1
new_state.categorie = gcat
new_state.club = club
new_state.country = country_
new_state.genre = genre
} }
setModalState(new_state) setModalState(new_state)
sendRegister(new_state) sendRegister(new_state)
}}> }}>
<div className="modal-header"> <div className="modal-header">
<h1 className="modal-title fs-5" id="registerLabel">{editMode ? "Modification d'" : "Ajouter "}un combattant</h1> <h1 className="modal-title fs-5"
id="registerLabel">{editMode ? "Modification d'" : "Ajouter "}un {modalState.id >= 0 ? "combattant" : "invité"}</h1>
<button type="button" className="btn-close" data-bs-dismiss="modal" aria-label="Close"></button> <button type="button" className="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div> </div>
<div className="modal-body"> <div className="modal-body">
{modalState.id < 0 && <div className="mb-2">Les invités sont réservés aux membres non licenciés par la fédération. Les combattants inscrits via ce formulaire ne pourront pas voir leur résultat depuis leur profil.</div>}
<div className="card" style={{marginBottom: "1em"}}> <div className="card" style={{marginBottom: "1em"}}>
<div className="card-header">Recherche*</div> <div className="card-header">{modalState.id >= 0 ? "Recherche*" : "Information"}</div>
<div className="card-body"> <div className="card-body">
<div className="row"> <div className="row" hidden={modalState.id < 0}>
<div className="col"> <div className="col">
<input type="number" min={0} step={1} className="form-control" placeholder="N° de licence" name="licence" <input type="number" min={0} step={1} className="form-control" placeholder="N° de licence" name="licence"
value={licence} onChange={e => setLicence(e.target.value)} disabled={editMode}/> value={licence} onChange={e => setLicence(e.target.value)} disabled={editMode}/>
</div> </div>
</div> </div>
<h5 style={{textAlign: "center", marginTop: "0.25em"}}>Ou</h5> <h5 style={{textAlign: "center", marginTop: "0.25em"}} hidden={modalState.id < 0}>Ou</h5>
<div className="row"> <div className="row">
<div className="col"> <div className="col">
<input type="text" className="form-control" placeholder="Prénom" name="fname" disabled={editMode} <input type="text" className="form-control" placeholder="Prénom" name="fname"
disabled={editMode && modalState.id >= 0}
value={fname} onChange={e => setFname(e.target.value)}/> value={fname} onChange={e => setFname(e.target.value)}/>
</div> </div>
<div className="col"> <div className="col">
<input type="text" className="form-control" placeholder="Nom" name="lname" disabled={editMode} <input type="text" className="form-control" placeholder="Nom" name="lname"
disabled={editMode && modalState.id >= 0}
value={lname} onChange={e => setLname(e.target.value)}/> value={lname} onChange={e => setLname(e.target.value)}/>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div className="input-group mb-3" hidden={modalState.id >= 0}>
<span className="input-group-text" id="categorie">Club</span>
<input type="text" className="form-control" placeholder="Club" name="club"
value={club} onChange={e => setClub(e.target.value)}/>
</div>
<div className="input-group mb-3" hidden={modalState.id >= 0}>
<span className="input-group-text" id="categorie">Catégorie</span>
<select id="inputState2" className="form-select" value={gcat}
onChange={(e) => setGCat(e.target.value)}>
<option>-- Sélectionner catégorie --</option>
{CatList.map((cat, index) => {
return (<option key={index} value={cat}>{getCatName(cat)}</option>)
})}
</select>
</div>
<div className="input-group mb-3" hidden={modalState.id >= 0}>
<span className="input-group-text" id="categorie">Pays</span>
<select id="inputState0" className="form-select" value={country_} onChange={(e) => setCountry_(e.target.value)}>
{country && Object.keys(country).sort((a, b) => {
if (a < b) return -1
if (a > b) return 1
return 0
}).map((key, _) => {
return (<option key={key} value={key}>{country[key]}</option>)
})}
</select>
</div>
<div className="input-group mb-3" hidden={modalState.id >= 0}>
<span className="input-group-text" id="categorie">Genre</span>
<select className="form-select" aria-label="categorie" name="categorie" value={genre}
onChange={e => setGenre(e.target.value)}>
<option value={"NA"}>NA</option>
<option value={"H"}>H</option>
<option value={"F"}>F</option>
</select>
</div>
<div className="input-group mb-3"> <div className="input-group mb-3">
<span className="input-group-text" id="weight">Poids (en kg)</span> <span className="input-group-text" id="weight">Poids (en kg)</span>
<input type="number" min={1} step={1} className="form-control" placeholder="42" aria-label="weight" <input type="number" min={1} step={1} className="form-control" placeholder="42" aria-label="weight"
name="weight" aria-describedby="weight" value={weight} onChange={e => setWeight(e.target.value)}/> name="weight" aria-describedby="weight" value={weight} onChange={e => setWeight(e.target.value)}/>
</div> </div>
<div className="input-group mb-3"> <div className="input-group mb-3" hidden={modalState.id < 0}>
<span className="input-group-text" id="categorie">Surclassement</span> <span className="input-group-text" id="categorie">Surclassement</span>
<select className="form-select" aria-label="categorie" name="categorie" value={cat} <select className="form-select" aria-label="categorie" name="categorie" value={cat}
onChange={e => setCat(Number(e.target.value))}> onChange={e => setCat(Number(e.target.value))}>
@ -360,7 +437,7 @@ function Modal({sendRegister, modalState, setModalState, source}) {
</select> </select>
</div> </div>
{editMode && source === "admin" && <div className="form-check form-switch form-check-reverse"> {editMode && source === "admin" && <div className="form-check form-switch form-check-reverse" hidden={modalState.id < 0}>
<input className="form-check-input" type="checkbox" id="switchCheckReverse" checked={lockEdit} <input className="form-check-input" type="checkbox" id="switchCheckReverse" checked={lockEdit}
onChange={e => setLockEdit(e.target.checked)}/> onChange={e => setLockEdit(e.target.checked)}/>
<label className="form-check-label" htmlFor="switchCheckReverse">Empêcher les membres/club de modifier cette <label className="form-check-label" htmlFor="switchCheckReverse">Empêcher les membres/club de modifier cette
@ -436,7 +513,7 @@ function MakeCentralPanel({data, dispatch, id, setModalState, source}) {
</div> </div>
<div className="row"> <div className="row">
<div className="col-auto" style={{textAlign: "right"}}> <div className="col-auto" style={{textAlign: "right"}}>
<small>{req.data.categorie + (req.data.overCategory === 0 ? "" : (" avec " + req.data.overCategory + " de surclassement"))}<br/> <small>{getCatName(req.data.categorie) + (req.data.overCategory === 0 ? "" : (" avec " + req.data.overCategory + " de surclassement"))}<br/>
{req.data.weight ? req.data.weight : "---"} kg {req.data.weight ? req.data.weight : "---"} kg
</small> </small>
</div> </div>

View File

@ -99,6 +99,8 @@ export function getCatName(cat) {
return "Vétéran 1"; return "Vétéran 1";
case "VETERAN2": case "VETERAN2":
return "Vétéran 2"; return "Vétéran 2";
case null:
return "Catégorie inconnue";
default: default:
return cat; return cat;
} }