diff --git a/pom.xml b/pom.xml index 78ee59d..4614946 100644 --- a/pom.xml +++ b/pom.xml @@ -36,6 +36,21 @@ + + org.springframework.boot + spring-boot-starter + + + org.springframework.boot + spring-boot-starter-logging + + + + + org.springframework.boot + spring-boot-starter-log4j2 + + jakarta.servlet jakarta.servlet-api @@ -55,6 +70,7 @@ lombok provided + org.junit.jupiter junit-jupiter-api @@ -65,6 +81,23 @@ junit-jupiter-engine test + + org.mockito + mockito-junit-jupiter + test + + + org.jetbrains + annotations + 17.0.0 + test + + + + com.fasterxml.jackson.dataformat + jackson-dataformat-yaml + + @@ -88,6 +121,14 @@ + + org.apache.maven.plugins + maven-surefire-plugin + 3.1.2 + + false + + diff --git a/src/main/java/com/javarush/khmelov/cmd/EditUser.java b/src/main/java/com/javarush/khmelov/cmd/EditUser.java deleted file mode 100644 index ae191b4..0000000 --- a/src/main/java/com/javarush/khmelov/cmd/EditUser.java +++ /dev/null @@ -1,52 +0,0 @@ -package com.javarush.khmelov.cmd; - -import com.javarush.khmelov.entity.Role; -import com.javarush.khmelov.entity.User; -import com.javarush.khmelov.service.UserService; -import jakarta.servlet.http.HttpServletRequest; - -import java.util.Optional; - - -@SuppressWarnings("unused") -public class EditUser implements Command { - - private final UserService userService; - - public EditUser(UserService userService) { - this.userService = userService; - } - - - @Override - public String doGet(HttpServletRequest req) { - String stringId = req.getParameter("id"); - if (stringId != null) { - long id = Long.parseLong(stringId); - Optional optionalUser = userService.get(id); - if (optionalUser.isPresent()) { - User user = optionalUser.get(); - req.setAttribute("user", user); - } - } - return getView(); - } - - @Override - public String doPost(HttpServletRequest req) { - User user = User.builder() - .login(req.getParameter("login")) - .password(req.getParameter("password")) - .role(Role.valueOf(req.getParameter("role"))) - .build(); - if (req.getParameter("create") != null) { - userService.create(user); - } else if (req.getParameter("update") != null) { - user.setId(Long.parseLong(req.getParameter("id"))); - userService.update(user); - } - return getView() + "?id=" + user.getId(); - } - - -} \ No newline at end of file diff --git a/src/main/java/com/javarush/khmelov/cmd/ListUser.java b/src/main/java/com/javarush/khmelov/cmd/ListUser.java deleted file mode 100644 index 9257917..0000000 --- a/src/main/java/com/javarush/khmelov/cmd/ListUser.java +++ /dev/null @@ -1,26 +0,0 @@ -package com.javarush.khmelov.cmd; - -import com.javarush.khmelov.entity.User; -import com.javarush.khmelov.service.UserService; -import jakarta.servlet.http.HttpServletRequest; - -import java.util.Collection; - -@SuppressWarnings("unused") -public class ListUser implements Command { - - private final UserService userService; - - public ListUser(UserService userService) { - this.userService = userService; - } - - @Override - public String doGet(HttpServletRequest request) { - Collection users = userService.getAll(); - request.setAttribute("users", users); - return getView(); - } - - -} \ No newline at end of file diff --git a/src/main/java/com/javarush/khmelov/cmd/StartPage.java b/src/main/java/com/javarush/khmelov/cmd/StartPage.java deleted file mode 100644 index d268f93..0000000 --- a/src/main/java/com/javarush/khmelov/cmd/StartPage.java +++ /dev/null @@ -1,6 +0,0 @@ -package com.javarush.khmelov.cmd; - -@SuppressWarnings("unused") -public class StartPage implements Command { - -} diff --git a/src/main/java/com/javarush/khmelov/repository/Repository.java b/src/main/java/com/javarush/khmelov/repository/Repository.java deleted file mode 100644 index f1abdac..0000000 --- a/src/main/java/com/javarush/khmelov/repository/Repository.java +++ /dev/null @@ -1,19 +0,0 @@ -package com.javarush.khmelov.repository; - -import com.javarush.khmelov.entity.User; - -import java.util.Collection; -import java.util.Optional; - -public interface Repository { - - Collection getAll(); - - Optional get(long id); - - void create(T entity); - - void update(T entity); - - void delete(T entity); -} diff --git a/src/main/java/com/javarush/khmelov/repository/UserRepository.java b/src/main/java/com/javarush/khmelov/repository/UserRepository.java deleted file mode 100644 index 58b32ea..0000000 --- a/src/main/java/com/javarush/khmelov/repository/UserRepository.java +++ /dev/null @@ -1,50 +0,0 @@ -package com.javarush.khmelov.repository; - -import com.javarush.khmelov.entity.Role; -import com.javarush.khmelov.entity.User; - -import java.util.Collection; -import java.util.HashMap; -import java.util.Map; -import java.util.Optional; -import java.util.concurrent.atomic.AtomicLong; - -public class UserRepository implements Repository { - - private final Map map = new HashMap<>(); - - public static final AtomicLong id = new AtomicLong(System.currentTimeMillis()); - - public UserRepository() { - map.put(1L, new User(1L, "Alisa", "qwerty", Role.USER)); - map.put(2L, new User(2L, "Bob", "", Role.GUEST)); - map.put(3L, new User(3L, "Carl", "admin", Role.ADMIN)); - map.put(4L, new User(4L, "Khmelov", "admin", Role.ADMIN)); - } - - @Override - public Collection getAll() { - return map.values(); - } - - @Override - public Optional get(long id) { - return Optional.ofNullable(map.get(id)); - } - - @Override - public void create(User entity) { - entity.setId(id.incrementAndGet()); - update(entity); - } - - @Override - public void update(User entity) { - map.put(entity.getId(), entity); - } - - @Override - public void delete(User entity) { - map.remove(entity.getId()); - } -} diff --git a/src/main/java/com/javarush/khmelov/service/UserService.java b/src/main/java/com/javarush/khmelov/service/UserService.java deleted file mode 100644 index b17527c..0000000 --- a/src/main/java/com/javarush/khmelov/service/UserService.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.javarush.khmelov.service; - -import com.javarush.khmelov.entity.User; -import com.javarush.khmelov.repository.UserRepository; - -import java.util.Collection; -import java.util.Optional; - -public class UserService { - - private final UserRepository userRepository; - - public UserService(UserRepository userRepository) { - this.userRepository = userRepository; - } - - public void create(User user) { - userRepository.create(user); - } - - public void update(User user) { - userRepository.update(user); - } - - public void delete(User user) { - userRepository.delete(user); - } - - public Collection getAll() { - return userRepository.getAll(); - } - - public Optional get(long id) { - return userRepository.get(id); - } -} diff --git a/src/main/java/com/javarush/khmelov/cmd/Command.java b/src/main/java/com/javarush/vasileva/cmd/Command.java similarity index 88% rename from src/main/java/com/javarush/khmelov/cmd/Command.java rename to src/main/java/com/javarush/vasileva/cmd/Command.java index fd4035b..d86b3a4 100644 --- a/src/main/java/com/javarush/khmelov/cmd/Command.java +++ b/src/main/java/com/javarush/vasileva/cmd/Command.java @@ -1,4 +1,4 @@ -package com.javarush.khmelov.cmd; +package com.javarush.vasileva.cmd; import jakarta.servlet.http.HttpServletRequest; @@ -15,6 +15,10 @@ default String doPost(HttpServletRequest request) { return getView(); } + default String doDelete(HttpServletRequest request) { + return getView(); + } + default String getView() { String simpleName = this.getClass().getSimpleName(); return convertCamelCaseToKebabStyle(simpleName); @@ -32,6 +36,4 @@ private static String convertCamelCaseToKebabStyle(String string) { ? snakeName.substring(1) : snakeName; } - - } diff --git a/src/main/java/com/javarush/vasileva/cmd/EditQuest.java b/src/main/java/com/javarush/vasileva/cmd/EditQuest.java new file mode 100644 index 0000000..1d69dde --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/EditQuest.java @@ -0,0 +1,101 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.config.Config; +import com.javarush.vasileva.config.Winter; +import com.javarush.vasileva.entity.Quest; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.mapper.QuestMapper; +import com.javarush.vasileva.service.AuthService; +import com.javarush.vasileva.service.QuestService; +import jakarta.servlet.http.HttpServletRequest; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; +import java.util.List; + +import static com.javarush.vasileva.util.Key.*; +import static com.javarush.vasileva.util.Link.HOME; +import static com.javarush.vasileva.util.Value.*; + +@SuppressWarnings("unused") +public class EditQuest implements Command { + + private static final Logger LOGGER = LoggerFactory.getLogger(EditQuest.class.getName()); + + private final QuestService questService; + private final AuthService authService; + private final QuestMapper questMapper = Winter.find(QuestMapper.class); + private final Config config; + + @SuppressWarnings("unused") + public EditQuest(QuestService questService, AuthService authService, Config config) { + this.questService = questService; + this.authService = authService; + this.config = config; + } + + @Override + public String doGet(HttpServletRequest req) { + LOGGER.info("Received GET request to edit quests"); + + authService.checkAdminAuthorization(req, EDIT_QUEST_AUTH_ERROR); + + List quests = questService.getAll(); + req.setAttribute(QUESTS, quests); + LOGGER.debug("Loaded quests: {}", quests.size()); + + String questIdStr = req.getParameter(QUEST_ID); + if (questIdStr != null && !questIdStr.isEmpty()) { + LOGGER.info("Editing quest with id: {}", questIdStr); + Quest quest = questService.getValidatedQuest(questIdStr) + .orElseThrow(() -> new AppException(QUEST_NOT_FOUND + questIdStr)); + req.setAttribute(EDIT, true); + try { + String questJson = questMapper.toJsonString(quest); + req.setAttribute(QUEST_JSON, questJson); + LOGGER.debug("Quest is mapped to JSON. ID: {}", quest.getId()); + } catch (IOException e) { + return getView(); + } + } else { + LOGGER.info("Creating quest"); + req.setAttribute(EDIT, false); + req.setAttribute(QUEST_JSON, JSON_SAMPLE); + } + return getView(); + } + + @Override + public String doPost(HttpServletRequest req) { + LOGGER.info("Processing POST-request for saving quest. Параметры: {}", req.getParameterMap()); + String questJson = req.getParameter(QUEST_JSON); + + if (questJson == null || questJson.isEmpty()) { + LOGGER.error(JSON_SAVE_ERROR + ". JSON parameter is missing or empty"); + req.getSession().setAttribute(ERROR, JSON_SAVE_ERROR); + return getView(); + } + + try { + Quest quest = questMapper.fromJsonString(questJson); + LOGGER.debug("Quest is successfully received from JSON. Title: {}", quest.getTitle()); + if (quest.getId() != null) { + LOGGER.info("Updating quest with ID: {}", quest.getId()); + questService.update(quest); + } else { + LOGGER.info("Creating new quest"); + questService.create(quest); + } + config.setQuestParameters(quest); + LOGGER.info("Quest is successfully saved. ID: {}", quest.getId()); + return HOME; + } catch (IOException e) { + LOGGER.error(JSON_SAVE_ERROR + ". JSON: {}. Cause: {}", + questJson, e.getMessage(), e); + req.getSession().setAttribute(QUEST_JSON, questJson); + req.getSession().setAttribute(ERROR, JSON_SAVE_ERROR); + return getView(); + } + } +} diff --git a/src/main/java/com/javarush/vasileva/cmd/EditUser.java b/src/main/java/com/javarush/vasileva/cmd/EditUser.java new file mode 100644 index 0000000..394c4a2 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/EditUser.java @@ -0,0 +1,70 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.entity.Role; +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.service.UserService; +import com.javarush.vasileva.util.Key; +import com.javarush.vasileva.util.Link; +import com.javarush.vasileva.util.Helpers; +import jakarta.servlet.http.HttpServletRequest; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Optional; + + +@SuppressWarnings("unused") +public class EditUser implements Command { + + private static final Logger LOGGER = LoggerFactory.getLogger(EditUser.class.getName()); + + private final UserService userService; + + public EditUser(UserService userService) { + this.userService = userService; + } + + @Override + public String doGet(HttpServletRequest req) { + String stringId = req.getParameter("id"); + LOGGER.info("Received GET request to edit user with id: {}", stringId); + + Optional optionalUser = userService.getValidatedUser(stringId); + + if (optionalUser.isPresent()) { + User user = optionalUser.get(); + req.setAttribute("user", user); + LOGGER.debug("User found and set as request attribute: {}", user.getLogin()); + } else { + LOGGER.warn("User with id {} not found", stringId); + } + + return getView(); + } + + @Override + public String doPost(HttpServletRequest req) { + LOGGER.info("Received POST request to process user data"); + + User user = User.builder() + .login(req.getParameter(Key.LOGIN)) + .email(req.getParameter(Key.EMAIL)) + .password(req.getParameter(Key.PASSWORD)) + .role(Role.valueOf(req.getParameter(Key.ROLE))) + .build(); + + LOGGER.debug("Constructed user object: login={}, email={}, role={}", + user.getLogin(), user.getEmail(), user.getRole()); + + if (req.getParameter(Key.CREATE) != null) { + LOGGER.info("Creating new user with login: {}", user.getLogin()); + userService.create(user); + } else if (req.getParameter(Key.UPDATE) != null) { + user.setId(Helpers.parseStringToLong(req.getParameter("id"))); + LOGGER.info("Updating user with id: {}, login: {}", user.getId(), user.getLogin()); + userService.update(user); + } + LOGGER.info("Redirecting to user list page"); + return Link.USER_LIST; + } +} \ No newline at end of file diff --git a/src/main/java/com/javarush/vasileva/cmd/ErrorPage.java b/src/main/java/com/javarush/vasileva/cmd/ErrorPage.java new file mode 100644 index 0000000..5f533dd --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/ErrorPage.java @@ -0,0 +1,5 @@ +package com.javarush.vasileva.cmd; + +@SuppressWarnings("unused") +public class ErrorPage implements Command{ +} diff --git a/src/main/java/com/javarush/vasileva/cmd/Home.java b/src/main/java/com/javarush/vasileva/cmd/Home.java new file mode 100644 index 0000000..15cb48c --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/Home.java @@ -0,0 +1,56 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.entity.Quest; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.AuthService; +import com.javarush.vasileva.service.QuestService; +import jakarta.servlet.http.HttpServletRequest; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +import static com.javarush.vasileva.util.Key.*; +import static com.javarush.vasileva.util.Value.*; + +@SuppressWarnings("unused") +public class Home implements Command { + + private static final Logger LOGGER = LoggerFactory.getLogger(Home.class.getName()); + + private final QuestService questService; + private final AuthService authService; + + public Home(QuestService questService, AuthService authService) { + this.questService = questService; + this.authService = authService; + } + + @Override + public String doGet(HttpServletRequest req) { + LOGGER.info("Received GET request to display all quests"); + List quests = questService.getAll(); + req.setAttribute(QUESTS, quests); + LOGGER.debug("Retrieved {} quests from service", quests.size()); + return getView(); + } + + @Override + public String doDelete(HttpServletRequest req) { + LOGGER.info("Received DELETE request for quest"); + + authService.checkAdminAuthorization(req, DELETE_QUEST_AUTH_ERROR); + LOGGER.debug("Admin authorization successful"); + + String questIdStr = req.getParameter(QUEST_ID); + LOGGER.debug("Attempting to delete quest with ID: {}", questIdStr); + + Quest quest = questService.getValidatedQuest(questIdStr) + .orElseThrow(() -> new AppException(QUEST_NOT_FOUND + questIdStr)); + req.setAttribute(QUEST, quest); + questService.delete(quest); + LOGGER.info("Quest with ID {} successfully deleted", questIdStr); + + return getView(); + } +} diff --git a/src/main/java/com/javarush/vasileva/cmd/Login.java b/src/main/java/com/javarush/vasileva/cmd/Login.java new file mode 100644 index 0000000..3e79450 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/Login.java @@ -0,0 +1,68 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.entity.UserStats; +import com.javarush.vasileva.service.UserService; +import com.javarush.vasileva.service.UserStatsService; +import com.javarush.vasileva.util.Key; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpSession; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Optional; + +import static com.javarush.vasileva.util.Key.*; +import static com.javarush.vasileva.util.Key.ERROR; +import static com.javarush.vasileva.util.Link.*; +import static com.javarush.vasileva.util.Value.*; + +@SuppressWarnings("unused") +public class Login implements Command { + + private static final Logger LOGGER = LoggerFactory.getLogger(Login.class.getName()); + + private final UserService userService; + private final UserStatsService userStatsService; + + public Login(UserService userService, UserStatsService userStatsService) { + this.userService = userService; + this.userStatsService = userStatsService; + } + + @Override + public String doPost(HttpServletRequest request) { + LOGGER.info("Received POST request for user login"); + + String email = request.getParameter(EMAIL); + String password = request.getParameter(PASSWORD); + + LOGGER.debug("Login attempt with email: {}, password: {}", email, password); + + if (email.isEmpty() || password.isEmpty()) { + LOGGER.warn("Empty email or password provided"); + request.getSession().setAttribute(Key.ERROR, EMPTY_DATA_ERROR); + return getView(); + } + + Optional optionalUser = userService.login(email, password); + if (optionalUser.isPresent()) { + User user = optionalUser.get(); + LOGGER.info("User authenticated successfully: {}", user.getEmail()); + + HttpSession session = request.getSession(); + UserStats stats = userStatsService.getUserStats(user.getId()) + .orElse(userStatsService.createUserStats(user.getId())); + + session.setAttribute(USER, optionalUser.get()); + session.setAttribute(Key.STATS, stats); + + LOGGER.debug("User stats loaded/created and stored in session for user ID: {}", user.getId()); + } else { + LOGGER.error(INVALID_DATA_ERROR); + request.getSession().setAttribute(ERROR, INVALID_DATA_ERROR); + return getView(); + } + return HOME; + } +} diff --git a/src/main/java/com/javarush/vasileva/cmd/Logout.java b/src/main/java/com/javarush/vasileva/cmd/Logout.java new file mode 100644 index 0000000..a5714a4 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/Logout.java @@ -0,0 +1,28 @@ +package com.javarush.vasileva.cmd; + +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpSession; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static com.javarush.vasileva.util.Link.LOGIN; + +@SuppressWarnings("unused") +public class Logout implements Command { + private static final Logger LOGGER = LoggerFactory.getLogger(Logout.class.getName()); + + @Override + public String doGet(HttpServletRequest request) { + LOGGER.info("Received GET request for user logout"); + + HttpSession session = request.getSession(false); + if (session != null) { + LOGGER.debug("User session found. Session ID: {}", session.getId()); + session.invalidate(); + LOGGER.info("User session invalidated successfully"); + } + + LOGGER.info("Redirecting user to login page"); + return LOGIN; + } +} diff --git a/src/main/java/com/javarush/vasileva/cmd/PlayGame.java b/src/main/java/com/javarush/vasileva/cmd/PlayGame.java new file mode 100644 index 0000000..3a5590b --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/PlayGame.java @@ -0,0 +1,88 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.entity.*; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.GameService; +import com.javarush.vasileva.util.Helpers; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpSession; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static com.javarush.vasileva.util.Key.*; +import static com.javarush.vasileva.util.Value.*; + +@SuppressWarnings("unused") +public class PlayGame implements Command { + + private static final Logger LOGGER = LoggerFactory.getLogger(PlayGame.class.getName()); + + private final GameService gameService; + + public PlayGame(GameService gameService) { + this.gameService = gameService; + } + + @Override + public String doGet(HttpServletRequest req) { + LOGGER.info("Received GET request to play game"); + + HttpSession session = req.getSession(); + User user = (User) session.getAttribute(USER); + + if (user == null) { + LOGGER.error("User not authenticated. Throwing AppException: {}", AUTH_ERROR); + throw new AppException(AUTH_ERROR); + } + LOGGER.debug("Authenticated user: id={}, login={}", user.getId(), user.getLogin()); + + Long questId = Helpers.parseStringToLong(req.getParameter(QUEST_ID)); + Long gameId = req.getParameter(GAME_ID) != null + ? Helpers.parseStringToLong(req.getParameter(GAME_ID)) + : null; + + LOGGER.debug("Quest ID: {}, Game ID: {}", questId, gameId); + + Game game; + if (gameId == null) { + LOGGER.info("Starting new game for quest ID: {}", questId); + game = gameService.startNewGame(questId, user.getId()); + } else { + LOGGER.info("Retrieving existing game by ID: {}", gameId); + game = gameService.getGameById(gameId).orElseThrow(() -> new AppException(GAME_NOT_FOUND)); + } + + req.setAttribute(GAME, game); + req.setAttribute(STATE, game.getGameState().isCompleted()); + req.setAttribute(QUEST, game.getGameState().getCurrentQuest()); + req.setAttribute(WINNING, game.getGameState().getCurrentQuestion().getLabel().contains(WIN)); + + LOGGER.debug("Game attributes set in request: gameId={}, completed={}, questId={}", + game.getId(), game.getGameState().isCompleted(), game.getGameState().getCurrentQuest().getId()); + + return getView(); + } + + @Override + public String doPost(HttpServletRequest req) throws AppException { + LOGGER.info("Received POST request to advance game"); + + HttpSession session = req.getSession(); + User user = (User) session.getAttribute(USER); + + Long gameId = Helpers.parseStringToLong(req.getParameter(GAME_ID)); + Long answerId = Helpers.parseStringToLong(req.getParameter(SELECTED_ANSWER_ID)); + + LOGGER.debug("Advancing game ID: {} with answer ID: {}", gameId, answerId); + Game updatedGame = gameService.advanceGame(gameId, answerId); + + req.setAttribute(GAME, updatedGame); + req.setAttribute(USER, user); + + String redirectUrl = getView() + "?" + GAME_ID + "=" + updatedGame.getId() + + "&" + QUEST_ID + "=" + updatedGame.getGameState().getCurrentQuest().getId(); + LOGGER.info("Game advanced successfully. Redirecting to: {}", redirectUrl); + + return redirectUrl; + } +} diff --git a/src/main/java/com/javarush/vasileva/cmd/Profile.java b/src/main/java/com/javarush/vasileva/cmd/Profile.java new file mode 100644 index 0000000..6b08a0e --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/Profile.java @@ -0,0 +1,5 @@ +package com.javarush.vasileva.cmd; + +@SuppressWarnings("unused") +public class Profile implements Command { +} diff --git a/src/main/java/com/javarush/vasileva/cmd/Register.java b/src/main/java/com/javarush/vasileva/cmd/Register.java new file mode 100644 index 0000000..882e0b0 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/Register.java @@ -0,0 +1,47 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.UserService; +import com.javarush.vasileva.util.Key; +import com.javarush.vasileva.util.Link; +import jakarta.servlet.http.HttpServletRequest; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static com.javarush.vasileva.util.Key.*; +import static com.javarush.vasileva.util.Value.*; + +@SuppressWarnings("unused") +public class Register implements Command { + + private static final Logger LOGGER = LoggerFactory.getLogger(Register.class.getName()); + + private final UserService userService; + + public Register(UserService userService) { + this.userService = userService; + } + + @Override + public String doPost(HttpServletRequest request) { + LOGGER.info("Received POST request for user registration"); + + String login = request.getParameter(Key.LOGIN); + String email = request.getParameter(EMAIL); + String password = request.getParameter(PASSWORD); + + LOGGER.debug("Registration attempt with login: {}, email: {}", login, email); + + if (login == null || login.isEmpty() || email == null || email.isEmpty() || password == null || password.isEmpty()) { + LOGGER.warn(EMPTY_DATA_ERROR + ". Missing: login={}, email={}, password={}", + login == null || login.isEmpty(), email == null || email.isEmpty(), password == null || password.isEmpty()); + throw new AppException(EMPTY_DATA_ERROR); + } + + LOGGER.info("Registering new user with login: {} and email: {}", login, email); + userService.register(login, email, password); + + LOGGER.info("User successfully registered. Redirecting to login page"); + return Link.LOGIN; + } +} diff --git a/src/main/java/com/javarush/vasileva/cmd/Stats.java b/src/main/java/com/javarush/vasileva/cmd/Stats.java new file mode 100644 index 0000000..0af9739 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/Stats.java @@ -0,0 +1,47 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.entity.UserStats; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.UserStatsService; +import com.javarush.vasileva.util.Key; +import com.javarush.vasileva.util.Value; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpSession; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static com.javarush.vasileva.util.Value.AUTH_ERROR; + +@SuppressWarnings("unused") +public class Stats implements Command { + + private static final Logger LOGGER = LoggerFactory.getLogger(Stats.class.getName()); + + private final UserStatsService statsService; + + public Stats(UserStatsService statsService) { + this.statsService = statsService; + } + + @Override + public String doGet(HttpServletRequest req) { + LOGGER.info("Received GET request to view user statistics"); + + HttpSession session = req.getSession(); + User user = (User) session.getAttribute("user"); + + if (user == null) { + LOGGER.error("User not authenticated. Throwing AppException: {}", AUTH_ERROR); + throw new AppException(AUTH_ERROR); + } + + LOGGER.debug("Authenticated user: id={}, login={}", user.getId(), user.getLogin()); + + UserStats stats = statsService.getUserStats(user.getId()).orElseThrow(() -> new AppException(Value.STATS_NOT_FOUND)); + req.setAttribute(Key.STATS, stats); + LOGGER.info("Statistics loaded successfully for user ID: {}", user.getId()); + + return getView(); + } +} diff --git a/src/main/java/com/javarush/vasileva/cmd/UserList.java b/src/main/java/com/javarush/vasileva/cmd/UserList.java new file mode 100644 index 0000000..1a6906b --- /dev/null +++ b/src/main/java/com/javarush/vasileva/cmd/UserList.java @@ -0,0 +1,61 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.AuthService; +import com.javarush.vasileva.service.UserService; +import jakarta.servlet.http.HttpServletRequest; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Collection; + +import static com.javarush.vasileva.util.Key.*; +import static com.javarush.vasileva.util.Value.*; + +@SuppressWarnings("unused") +public class UserList implements Command { + + private static final Logger LOGGER = LoggerFactory.getLogger(UserList.class.getName()); + + private final UserService userService; + private final AuthService authService; + + public UserList(UserService userService, AuthService authService) { + this.userService = userService; + this.authService = authService; + } + + @Override + public String doGet(HttpServletRequest request) { + LOGGER.info("Received GET request to view user list"); + + authService.checkAdminAuthorization(request, USER_LIST_AUTH_ERROR); + LOGGER.debug("Admin authorization successful"); + + Collection users = userService.getAll(); + LOGGER.info("Retrieved {} users from database", users.size()); + + request.setAttribute(USERS, users); + LOGGER.debug("Users and admin role set in request attributes"); + + return getView(); + } + + @Override + public String doDelete(HttpServletRequest req) { + LOGGER.info("Received DELETE request to remove user"); + + String userIdStr = req.getParameter(USER_ID); + LOGGER.debug("Attempting to delete user with ID: {}", userIdStr); + + User user = userService.getValidatedUser(userIdStr) + .orElseThrow(() -> new AppException(USER_NOT_FOUND + userIdStr)); + LOGGER.info("User found for deletion: id={}, login={}", user.getId(), user.getLogin()); + + userService.delete(user); + LOGGER.info("User with ID {} successfully deleted", user.getId()); + + return getView(); + } +} \ No newline at end of file diff --git a/src/main/java/com/javarush/vasileva/config/Config.java b/src/main/java/com/javarush/vasileva/config/Config.java new file mode 100644 index 0000000..8f9e4cc --- /dev/null +++ b/src/main/java/com/javarush/vasileva/config/Config.java @@ -0,0 +1,72 @@ +package com.javarush.vasileva.config; + +import com.javarush.vasileva.entity.*; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.mapper.QuestMapper; +import com.javarush.vasileva.service.AnswerService; +import com.javarush.vasileva.service.QuestService; +import com.javarush.vasileva.service.QuestionService; +import com.javarush.vasileva.service.UserService; +import lombok.AllArgsConstructor; + +import java.io.IOException; +import java.util.List; + +import static com.javarush.vasileva.util.Key.*; +import static com.javarush.vasileva.util.Value.JSON_SAVE_ERROR; + +@AllArgsConstructor +public class Config { + private final UserService userService = Winter.find(UserService.class); + private final QuestService questService = Winter.find(QuestService.class); + private final QuestionService questionService = Winter.find(QuestionService.class); + private final AnswerService answerService = Winter.find(AnswerService.class); + private final QuestMapper questMapper = Winter.find(QuestMapper.class); + + public static final String[] QUEST_FILES = {QUEST_FILE_NAME_1, QUEST_FILE_NAME_2, QUEST_FILE_NAME_3}; + + public void fillRepository() { + try { + for (String fileName : QUEST_FILES) { + Quest quest = questMapper.readFromJson(fileName); + questService.create(quest); + setQuestParameters(quest); + } + userService.create(buildUser("Carl", "admin@gmail.com", "admin", Role.ADMIN)); + userService.create(buildUser("Alisa", "alisa@gmail.com", "qwerty", Role.USER)); + userService.create(buildUser("Bob", "bob@gmail.com", "12345", Role.USER)); + } catch (IOException e) { + System.out.println(e.getMessage()); + } + } + + private User buildUser(String login, String email, String password, Role role) { + return User.builder() + .login(login) + .email(email) + .password(password) + .role(role) + .build(); + } + + public void setQuestParameters(Quest quest) { + List questions = quest.getQuestions(); + if (questions == null || questions.isEmpty()) { + throw new AppException(JSON_SAVE_ERROR); + } + for (Question question : questions) { + question.setQuestId(quest.getId()); + questionService.create(question); + List answers = question.getAnswers(); + if (answers == null) { + continue; + } + for (Answer answer : answers) { + answer.setQuestionId(question.getGeneratedId()); + answerService.create(answer); + } + } + long startQuestionId = quest.getQuestions().get(0).getGeneratedId(); + quest.setStartQuestionId(startQuestionId); + } +} diff --git a/src/main/java/com/javarush/khmelov/config/Winter.java b/src/main/java/com/javarush/vasileva/config/Winter.java similarity index 96% rename from src/main/java/com/javarush/khmelov/config/Winter.java rename to src/main/java/com/javarush/vasileva/config/Winter.java index 48bd8a7..53de644 100644 --- a/src/main/java/com/javarush/khmelov/config/Winter.java +++ b/src/main/java/com/javarush/vasileva/config/Winter.java @@ -1,4 +1,4 @@ -package com.javarush.khmelov.config; +package com.javarush.vasileva.config; import lombok.SneakyThrows; diff --git a/src/main/java/com/javarush/khmelov/controller/FrontController.java b/src/main/java/com/javarush/vasileva/controller/FrontController.java similarity index 54% rename from src/main/java/com/javarush/khmelov/controller/FrontController.java rename to src/main/java/com/javarush/vasileva/controller/FrontController.java index 33242b2..86290d4 100644 --- a/src/main/java/com/javarush/khmelov/controller/FrontController.java +++ b/src/main/java/com/javarush/vasileva/controller/FrontController.java @@ -1,8 +1,9 @@ -package com.javarush.khmelov.controller; +package com.javarush.vasileva.controller; -import com.javarush.khmelov.cmd.Command; -import com.javarush.khmelov.config.Winter; -import com.javarush.khmelov.entity.Role; +import com.javarush.vasileva.cmd.Command; +import com.javarush.vasileva.config.Config; +import com.javarush.vasileva.config.Winter; +import com.javarush.vasileva.entity.Role; import jakarta.servlet.ServletConfig; import jakarta.servlet.ServletException; import jakarta.servlet.annotation.WebServlet; @@ -12,11 +13,21 @@ import java.io.IOException; -@WebServlet({"", "/home", "/list-user", "/edit-user"}) +import static com.javarush.vasileva.util.Link.*; + +@WebServlet({INDEX, HOME, LOGIN, REGISTER, LOGOUT, PROFILE, USER_LIST, EDIT_USER, PLAY_GAME, EDIT_QUEST, STATS, ERROR}) public class FrontController extends HttpServlet { private final HttpResolver httpResolver = Winter.find(HttpResolver.class); + @Override + public void init(ServletConfig config) { + config.getServletContext().setAttribute("roles", Role.values()); + + Config gameConfig = Winter.find(Config.class); + gameConfig.fillRepository(); + } + @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { Command command = httpResolver.resolve(req); @@ -25,19 +36,26 @@ protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws Se req.getRequestDispatcher(jsp).forward(req, resp); } - @Override - public void init(ServletConfig config) { - config.getServletContext().setAttribute("roles", Role.values()); - } - private static String getJsp(String view) { - return "/WEB-INF/" + view + ".jsp"; + return "WEB-INF/" + view + ".jsp"; } @Override - protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { + protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws IOException { + String methodOverride = req.getParameter("_method"); + if ("DELETE".equalsIgnoreCase(methodOverride)) { + doDelete(req, resp); + return; + } Command command = httpResolver.resolve(req); String redirect = command.doPost(req); resp.sendRedirect(redirect); } + + @Override + protected void doDelete(HttpServletRequest req, HttpServletResponse resp) throws IOException { + Command command = httpResolver.resolve(req); + String redirect = command.doDelete(req); + resp.sendRedirect(redirect); + } } diff --git a/src/main/java/com/javarush/khmelov/controller/HttpResolver.java b/src/main/java/com/javarush/vasileva/controller/HttpResolver.java similarity index 83% rename from src/main/java/com/javarush/khmelov/controller/HttpResolver.java rename to src/main/java/com/javarush/vasileva/controller/HttpResolver.java index 18bb761..175faf6 100644 --- a/src/main/java/com/javarush/khmelov/controller/HttpResolver.java +++ b/src/main/java/com/javarush/vasileva/controller/HttpResolver.java @@ -1,16 +1,16 @@ -package com.javarush.khmelov.controller; +package com.javarush.vasileva.controller; -import com.javarush.khmelov.cmd.Command; -import com.javarush.khmelov.config.Winter; +import com.javarush.vasileva.cmd.Command; +import com.javarush.vasileva.config.Winter; +import com.javarush.vasileva.util.Link; import jakarta.servlet.http.HttpServletRequest; public class HttpResolver { public Command resolve(HttpServletRequest request) { - // /cmd-example try { String requestURI = request.getRequestURI(); - requestURI = requestURI.equals("/") ? "/start-page" : requestURI; + requestURI = requestURI.equals("/") ? Link.HOME : requestURI; String kebabName = requestURI.split("[?#/]")[1]; String simpleName = convertKebabStyleToCamelCase(kebabName); String fullName = Command.class.getPackageName() + "." + simpleName; diff --git a/src/main/java/com/javarush/vasileva/entity/Answer.java b/src/main/java/com/javarush/vasileva/entity/Answer.java new file mode 100644 index 0000000..cda4e95 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/entity/Answer.java @@ -0,0 +1,18 @@ +package com.javarush.vasileva.entity; + +import lombok.*; + +@Getter +@Setter +@ToString +@EqualsAndHashCode +@AllArgsConstructor +@NoArgsConstructor +@Builder +public class Answer { + private Long id; + private Long questionId; + private String nextQuestionLabel; + private String text; + private String description; +} diff --git a/src/main/java/com/javarush/vasileva/entity/Game.java b/src/main/java/com/javarush/vasileva/entity/Game.java new file mode 100644 index 0000000..a9f5cb2 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/entity/Game.java @@ -0,0 +1,18 @@ +package com.javarush.vasileva.entity; + +import lombok.AllArgsConstructor; +import lombok.Builder; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@NoArgsConstructor +@AllArgsConstructor +@Builder +public class Game { + private Long id; + private Long questId; + private Long userId; + private Long currentQuestionId; + private GameState gameState; +} diff --git a/src/main/java/com/javarush/vasileva/entity/GameState.java b/src/main/java/com/javarush/vasileva/entity/GameState.java new file mode 100644 index 0000000..4dc3605 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/entity/GameState.java @@ -0,0 +1,17 @@ +package com.javarush.vasileva.entity; + +import lombok.AllArgsConstructor; +import lombok.Builder; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +@Builder +public class GameState { + private Quest currentQuest; + private Question currentQuestion; + private User user; + private boolean isCompleted; +} diff --git a/src/main/java/com/javarush/vasileva/entity/Quest.java b/src/main/java/com/javarush/vasileva/entity/Quest.java new file mode 100644 index 0000000..8973bec --- /dev/null +++ b/src/main/java/com/javarush/vasileva/entity/Quest.java @@ -0,0 +1,22 @@ +package com.javarush.vasileva.entity; + +import lombok.AllArgsConstructor; +import lombok.Builder; +import lombok.Data; +import lombok.NoArgsConstructor; + +import java.util.List; + +@Data +@NoArgsConstructor +@AllArgsConstructor +@Builder +public class Quest { + private Long id; + private String title; + private String description; + private String text; + private Long startQuestionId; + private List questions; + private String image; +} diff --git a/src/main/java/com/javarush/vasileva/entity/Question.java b/src/main/java/com/javarush/vasileva/entity/Question.java new file mode 100644 index 0000000..93abc46 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/entity/Question.java @@ -0,0 +1,20 @@ +package com.javarush.vasileva.entity; + +import lombok.AllArgsConstructor; +import lombok.Builder; +import lombok.Data; +import lombok.NoArgsConstructor; + +import java.util.List; + +@Data +@NoArgsConstructor +@AllArgsConstructor +@Builder +public class Question { + private Long generatedId; + private String label; + private Long questId; + private String text; + private List answers; +} diff --git a/src/main/java/com/javarush/khmelov/entity/Role.java b/src/main/java/com/javarush/vasileva/entity/Role.java similarity index 54% rename from src/main/java/com/javarush/khmelov/entity/Role.java rename to src/main/java/com/javarush/vasileva/entity/Role.java index 5ae365f..5c6bacf 100644 --- a/src/main/java/com/javarush/khmelov/entity/Role.java +++ b/src/main/java/com/javarush/vasileva/entity/Role.java @@ -1,4 +1,4 @@ -package com.javarush.khmelov.entity; +package com.javarush.vasileva.entity; public enum Role { USER, ADMIN, GUEST diff --git a/src/main/java/com/javarush/khmelov/entity/User.java b/src/main/java/com/javarush/vasileva/entity/User.java similarity index 62% rename from src/main/java/com/javarush/khmelov/entity/User.java rename to src/main/java/com/javarush/vasileva/entity/User.java index f7fa2d6..65c778b 100644 --- a/src/main/java/com/javarush/khmelov/entity/User.java +++ b/src/main/java/com/javarush/vasileva/entity/User.java @@ -1,4 +1,4 @@ -package com.javarush.khmelov.entity; +package com.javarush.vasileva.entity; import lombok.AllArgsConstructor; import lombok.Builder; @@ -15,12 +15,14 @@ public class User { private String login; + private String email; + private String password; private Role role; - public String getImage() { //TODO move to DTO - return "image-" + id; + @SuppressWarnings("unused") + public boolean isAdmin() { + return role != null && role.equals(Role.ADMIN); } - } diff --git a/src/main/java/com/javarush/vasileva/entity/UserStats.java b/src/main/java/com/javarush/vasileva/entity/UserStats.java new file mode 100644 index 0000000..cd23711 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/entity/UserStats.java @@ -0,0 +1,18 @@ +package com.javarush.vasileva.entity; + +import lombok.AllArgsConstructor; +import lombok.Builder; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@NoArgsConstructor +@AllArgsConstructor +@Builder +public class UserStats { + private long id; + private long userId; + private int total; + private int wins; + private int losses; +} diff --git a/src/main/java/com/javarush/vasileva/exception/AppException.java b/src/main/java/com/javarush/vasileva/exception/AppException.java new file mode 100644 index 0000000..36c445f --- /dev/null +++ b/src/main/java/com/javarush/vasileva/exception/AppException.java @@ -0,0 +1,7 @@ +package com.javarush.vasileva.exception; + +public class AppException extends RuntimeException { + public AppException(String message) { + super(message); + } +} diff --git a/src/main/java/com/javarush/vasileva/exception/ExceptionHelper.java b/src/main/java/com/javarush/vasileva/exception/ExceptionHelper.java new file mode 100644 index 0000000..a6558a8 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/exception/ExceptionHelper.java @@ -0,0 +1,13 @@ +package com.javarush.vasileva.exception; + +import com.javarush.vasileva.util.Key; +import jakarta.servlet.http.HttpServletRequest; + +public class ExceptionHelper { + private ExceptionHelper() { + } + + public static void createError(HttpServletRequest req, String errorMessage) { + req.getSession().setAttribute(Key.ERROR, errorMessage); + } +} diff --git a/src/main/java/com/javarush/vasileva/filter/CleanerFilter.java b/src/main/java/com/javarush/vasileva/filter/CleanerFilter.java new file mode 100644 index 0000000..4014713 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/filter/CleanerFilter.java @@ -0,0 +1,26 @@ +package com.javarush.vasileva.filter; + +import com.javarush.vasileva.util.Key; +import jakarta.servlet.FilterChain; +import jakarta.servlet.ServletException; +import jakarta.servlet.annotation.WebFilter; +import jakarta.servlet.http.HttpFilter; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpServletResponse; +import jakarta.servlet.http.HttpSession; + +import java.io.IOException; + +@WebFilter("/*") +public class CleanerFilter extends HttpFilter { + + @Override + protected void doFilter(HttpServletRequest req, HttpServletResponse res, FilterChain chain) throws IOException, ServletException { + chain.doFilter(req, res); + HttpSession session = req.getSession(false); + if(req.getMethod().equals("GET") && session != null) { + session.removeAttribute(Key.ERROR); + session.removeAttribute(Key.QUEST_JSON); + } + } +} diff --git a/src/main/java/com/javarush/vasileva/filter/EncoderFilter.java b/src/main/java/com/javarush/vasileva/filter/EncoderFilter.java new file mode 100644 index 0000000..2eb10be --- /dev/null +++ b/src/main/java/com/javarush/vasileva/filter/EncoderFilter.java @@ -0,0 +1,22 @@ +package com.javarush.vasileva.filter; + +import jakarta.servlet.FilterChain; +import jakarta.servlet.ServletException; +import jakarta.servlet.ServletRequest; +import jakarta.servlet.ServletResponse; +import jakarta.servlet.annotation.WebFilter; +import jakarta.servlet.http.HttpFilter; + +import java.io.IOException; +import java.nio.charset.StandardCharsets; + +@WebFilter({"/*", ""}) +public class EncoderFilter extends HttpFilter { + + @Override + public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException { + req.setCharacterEncoding(StandardCharsets.UTF_8.name()); + res.setCharacterEncoding(StandardCharsets.UTF_8.name()); + chain.doFilter(req, res); + } +} diff --git a/src/main/java/com/javarush/vasileva/filter/ExceptionFilter.java b/src/main/java/com/javarush/vasileva/filter/ExceptionFilter.java new file mode 100644 index 0000000..eef9788 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/filter/ExceptionFilter.java @@ -0,0 +1,33 @@ +package com.javarush.vasileva.filter; + +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.exception.ExceptionHelper; +import com.javarush.vasileva.util.Link; +import jakarta.servlet.FilterChain; +import jakarta.servlet.ServletException; +import jakarta.servlet.annotation.WebFilter; +import jakarta.servlet.http.HttpFilter; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpServletResponse; + +import java.io.IOException; + +@WebFilter("/*") +public class ExceptionFilter extends HttpFilter { + + @Override + protected void doFilter(HttpServletRequest req, HttpServletResponse res, FilterChain chain) + throws IOException, ServletException { + try { + chain.doFilter(req, res); + } catch (AppException e) { + handleAppException(req, res, e); + } + } + + private void handleAppException(HttpServletRequest req, HttpServletResponse res, AppException appException) + throws ServletException, IOException { + ExceptionHelper.createError(req, appException.getMessage()); + req.getRequestDispatcher(Link.ERROR).forward(req, res); + } +} diff --git a/src/main/java/com/javarush/vasileva/game/GameEngine.java b/src/main/java/com/javarush/vasileva/game/GameEngine.java new file mode 100644 index 0000000..4ecd7db --- /dev/null +++ b/src/main/java/com/javarush/vasileva/game/GameEngine.java @@ -0,0 +1,46 @@ +package com.javarush.vasileva.game; + +import com.javarush.vasileva.entity.*; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.AnswerService; +import com.javarush.vasileva.service.QuestionService; +import com.javarush.vasileva.service.UserStatsService; +import com.javarush.vasileva.util.Value; + +import static com.javarush.vasileva.util.Value.*; + +public class GameEngine { + private final QuestionService questionService; + private final AnswerService answerService; + private final UserStatsService userStatsService; + + public GameEngine(QuestionService questionService, AnswerService answerService, UserStatsService userStatsService) { + this.questionService = questionService; + this.answerService = answerService; + this.userStatsService = userStatsService; + } + + public GameState startGame(User user, Quest quest) { + Question startQuestion = questionService.findById(quest.getStartQuestionId()) + .orElseThrow(() -> new AppException(QUESTION_NOT_FOUND)); + boolean isFinalQuestion = questionService.isFinalQuestion(startQuestion); + + return new GameState(quest, startQuestion, user, isFinalQuestion); + } + + public GameState advanceGame(GameState currentState, Long answerId) { + Answer answer = answerService.findById(answerId).orElseThrow(() -> new AppException(ANSWER_NOT_FOUND)); + String nextQuestionLabel = answer.getNextQuestionLabel(); + Question nextQuestion = questionService.getByQuestionLabelAndQuestId(nextQuestionLabel, currentState.getCurrentQuest().getId()) + .orElseThrow(() -> new AppException(Value.QUESTION_NOT_FOUND + nextQuestionLabel)); + boolean isFinalQuestion = questionService.isFinalQuestion(nextQuestion); + + if (isFinalQuestion) { + UserStats stats = userStatsService.getUserStats(currentState.getUser().getId()) + .orElseThrow(() -> new AppException(STATS_NOT_FOUND)); + userStatsService.updateUserStats(nextQuestion, stats); + } + + return new GameState(currentState.getCurrentQuest(), nextQuestion, currentState.getUser(), isFinalQuestion); + } +} diff --git a/src/main/java/com/javarush/vasileva/mapper/QuestMapper.java b/src/main/java/com/javarush/vasileva/mapper/QuestMapper.java new file mode 100644 index 0000000..5d4a33c --- /dev/null +++ b/src/main/java/com/javarush/vasileva/mapper/QuestMapper.java @@ -0,0 +1,33 @@ +package com.javarush.vasileva.mapper; + +import com.fasterxml.jackson.databind.ObjectMapper; +import com.javarush.vasileva.entity.Quest; + +import java.io.IOException; +import java.io.InputStream; + +public class QuestMapper { + private final ObjectMapper objectMapper; + + public QuestMapper() { + this.objectMapper = new ObjectMapper(); + } + + public Quest readFromJson(String filePath) throws IOException { + InputStream inputStream = getClass().getResourceAsStream(filePath); + if (inputStream == null) { + throw new IOException("Ресурс не найден: " + filePath); + } + try (inputStream) { + return objectMapper.readValue(inputStream, Quest.class); + } + } + + public Quest fromJsonString(String jsonString) throws IOException { + return objectMapper.readValue(jsonString, Quest.class); + } + + public String toJsonString(Quest quest) throws IOException { + return objectMapper.writeValueAsString(quest); + } +} diff --git a/src/main/java/com/javarush/vasileva/repository/AnswerRepository.java b/src/main/java/com/javarush/vasileva/repository/AnswerRepository.java new file mode 100644 index 0000000..c668bc3 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/repository/AnswerRepository.java @@ -0,0 +1,39 @@ +package com.javarush.vasileva.repository; + +import com.javarush.vasileva.entity.Answer; + +import java.util.ArrayList; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.atomic.AtomicLong; + +public class AnswerRepository implements Repository { + private final Map answers = new ConcurrentHashMap<>(); + private static final AtomicLong generatedId = new AtomicLong(); + + @Override + public List getAll() { + return new ArrayList<>(answers.values()); + } + + @Override + public Optional findById(long id) { + return Optional.ofNullable(answers.get(id)); + } + + @Override + public void create(Answer answer) { + answer.setId(generatedId.incrementAndGet()); + answers.put(answer.getId(), answer); + } + + @Override + public void update(Answer entity) { + } + + @Override + public void delete(Answer entity) { + } +} diff --git a/src/main/java/com/javarush/vasileva/repository/GameRepository.java b/src/main/java/com/javarush/vasileva/repository/GameRepository.java new file mode 100644 index 0000000..bc33e31 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/repository/GameRepository.java @@ -0,0 +1,14 @@ +package com.javarush.vasileva.repository; + +import com.javarush.vasileva.entity.Game; + +import java.util.Optional; + +public interface GameRepository { + + Game save(Game game); + + Optional findById(long id); + + void delete(Game game); +} diff --git a/src/main/java/com/javarush/vasileva/repository/InMemoryGameRepository.java b/src/main/java/com/javarush/vasileva/repository/InMemoryGameRepository.java new file mode 100644 index 0000000..d3db2f6 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/repository/InMemoryGameRepository.java @@ -0,0 +1,35 @@ +package com.javarush.vasileva.repository; + +import com.javarush.vasileva.entity.Game; + +import java.util.Map; +import java.util.Optional; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.atomic.AtomicLong; + +public class InMemoryGameRepository implements GameRepository { + + private final Map map = new ConcurrentHashMap<>(); + private final AtomicLong generatedId = new AtomicLong(0); + + @Override + public Game save(Game game) { + if (game.getId() == null) { + game.setId(generatedId.incrementAndGet()); + } + map.put(game.getId(), game); + return game; + } + + @Override + public Optional findById(long id) { + return Optional.ofNullable(map.get(id)); + } + + @Override + public void delete(Game game) { + if (game != null && game.getId() != null) { + map.remove(game.getId()); + } + } +} diff --git a/src/main/java/com/javarush/vasileva/repository/QuestRepository.java b/src/main/java/com/javarush/vasileva/repository/QuestRepository.java new file mode 100644 index 0000000..39e5576 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/repository/QuestRepository.java @@ -0,0 +1,42 @@ +package com.javarush.vasileva.repository; + +import com.javarush.vasileva.entity.Quest; +import lombok.NoArgsConstructor; + +import java.util.*; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.atomic.AtomicLong; + +@NoArgsConstructor +public class QuestRepository implements Repository { + + private final Map quests = new ConcurrentHashMap<>(); + private final AtomicLong generatedId = new AtomicLong(10); + + @Override + public List getAll() { + return new ArrayList<>(quests.values()); + } + + @Override + public Optional findById(long id) { + return Optional.ofNullable(quests.get(id)); + } + + @Override + public void create(Quest quest) { + quest.setId(generatedId.incrementAndGet()); + quests.put(quest.getId(), quest); + } + + @Override + public void update(Quest quest) { + long questId = quest.getId(); + quests.put(questId, quest); + } + + @Override + public void delete(Quest quest) { + quests.remove(quest.getId()); + } +} diff --git a/src/main/java/com/javarush/vasileva/repository/QuestionRepository.java b/src/main/java/com/javarush/vasileva/repository/QuestionRepository.java new file mode 100644 index 0000000..14eeb7e --- /dev/null +++ b/src/main/java/com/javarush/vasileva/repository/QuestionRepository.java @@ -0,0 +1,49 @@ +package com.javarush.vasileva.repository; + +import com.javarush.vasileva.entity.Question; +import lombok.NoArgsConstructor; + +import java.util.ArrayList; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.atomic.AtomicLong; + +@NoArgsConstructor +public class QuestionRepository implements Repository { + private final Map questions = new ConcurrentHashMap<>(); + private static final AtomicLong generatedId = new AtomicLong(0); + + @Override + public List getAll() { + return new ArrayList<>(questions.values()); + } + + @Override + public Optional findById(long id) { + return Optional.ofNullable(questions.get(id)); // generatedId + } + + public Optional getByQuestionLabelAndQuestId(String questionLabel, long questId) { + return questions.values().stream() + .filter(question -> question.getLabel().equals(questionLabel)) + .filter(question -> question.getQuestId() == questId) + .findFirst(); + } + + @Override + public void create(Question question) { + question.setGeneratedId(generatedId.incrementAndGet()); + questions.put(question.getGeneratedId(), question); + } + + @Override + public void update(Question entity) { + + } + + @Override + public void delete(Question entity) { + } +} diff --git a/src/main/java/com/javarush/vasileva/repository/Repository.java b/src/main/java/com/javarush/vasileva/repository/Repository.java new file mode 100644 index 0000000..a178bd3 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/repository/Repository.java @@ -0,0 +1,17 @@ +package com.javarush.vasileva.repository; + +import java.util.List; +import java.util.Optional; + +public interface Repository { + + List getAll(); + + Optional findById(long id); + + void create(T entity); + + void update(T entity); + + void delete(T entity); +} diff --git a/src/main/java/com/javarush/vasileva/repository/UserRepository.java b/src/main/java/com/javarush/vasileva/repository/UserRepository.java new file mode 100644 index 0000000..7155da1 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/repository/UserRepository.java @@ -0,0 +1,45 @@ +package com.javarush.vasileva.repository; + +import com.javarush.vasileva.entity.User; + +import java.util.*; +import java.util.concurrent.atomic.AtomicLong; + +public class UserRepository implements Repository { + + private final Map map = new HashMap<>(); + + public static final AtomicLong id = new AtomicLong(System.currentTimeMillis()); + + @Override + public List getAll() { + return new ArrayList<>(map.values()); + } + + @Override + public Optional findById(long id) { + return Optional.ofNullable(map.get(id)); + } + + public Optional findByEmail(String email) { + return map.values().stream() + .filter(user -> user.getEmail().equals(email)) + .findFirst(); + } + + @Override + public void create(User user) { + user.setId(id.incrementAndGet()); + update(user); + } + + @Override + public void update(User entity) { + map.put(entity.getId(), entity); + } + + @Override + public void delete(User entity) { + map.remove(entity.getId()); + } +} diff --git a/src/main/java/com/javarush/vasileva/repository/UserStatsRepository.java b/src/main/java/com/javarush/vasileva/repository/UserStatsRepository.java new file mode 100644 index 0000000..a5efd5d --- /dev/null +++ b/src/main/java/com/javarush/vasileva/repository/UserStatsRepository.java @@ -0,0 +1,36 @@ +package com.javarush.vasileva.repository; + +import com.javarush.vasileva.entity.UserStats; + +import java.util.Map; +import java.util.Optional; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.atomic.AtomicLong; + +public class UserStatsRepository { + private final Map map = new ConcurrentHashMap<>(); + private final AtomicLong generatedId = new AtomicLong(0); + + public UserStats createUserStats(long userId) { + UserStats userStats = UserStats.builder() + .id(generatedId.incrementAndGet()) + .userId(userId) + .wins(0) + .losses(0) + .build(); + map.put(userStats.getId(), userStats); + return userStats; + } + + public Optional getUserStats(long userId) { + UserStats stats = map.values().stream() + .filter(userStats -> userStats.getUserId() == userId) + .findFirst() + .orElse(null); + return Optional.ofNullable(stats); + } + + public void updateUserStats(UserStats userStats) { + map.put(userStats.getId(), userStats); + } +} diff --git a/src/main/java/com/javarush/vasileva/service/AnswerService.java b/src/main/java/com/javarush/vasileva/service/AnswerService.java new file mode 100644 index 0000000..9e4514f --- /dev/null +++ b/src/main/java/com/javarush/vasileva/service/AnswerService.java @@ -0,0 +1,28 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Answer; +import com.javarush.vasileva.repository.AnswerRepository; +import com.javarush.vasileva.repository.Repository; + +import java.util.List; +import java.util.Optional; + +public class AnswerService { + private final Repository answerRepository; + + public AnswerService(AnswerRepository answerRepository) { + this.answerRepository = answerRepository; + } + + public List getAll() { + return answerRepository.getAll(); + } + + public Optional findById(Long id) { + return answerRepository.findById(id); + } + + public void create(Answer answer) { + answerRepository.create(answer); + } +} diff --git a/src/main/java/com/javarush/vasileva/service/AuthService.java b/src/main/java/com/javarush/vasileva/service/AuthService.java new file mode 100644 index 0000000..2bae593 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/service/AuthService.java @@ -0,0 +1,20 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Role; +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.exception.AppException; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpSession; + +import static com.javarush.vasileva.util.Key.USER; + +public class AuthService { + + public void checkAdminAuthorization(HttpServletRequest req, String message) { + HttpSession session = req.getSession(false); + User user = (User) session.getAttribute(USER); + if (user == null || user.getRole() == Role.GUEST || user.getRole() == Role.USER) { + throw new AppException(message); + } + } +} diff --git a/src/main/java/com/javarush/vasileva/service/GameService.java b/src/main/java/com/javarush/vasileva/service/GameService.java new file mode 100644 index 0000000..3535ae6 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/service/GameService.java @@ -0,0 +1,55 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Game; +import com.javarush.vasileva.entity.Quest; +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.game.GameEngine; +import com.javarush.vasileva.entity.GameState; +import com.javarush.vasileva.repository.InMemoryGameRepository; + +import java.util.Optional; + +import static com.javarush.vasileva.util.Value.*; + +public class GameService { + private final InMemoryGameRepository gameRepository; + private final UserService userService; + private final QuestService questService; + private final GameEngine gameEngine; + + public GameService(InMemoryGameRepository gameRepository, + UserService userService, + QuestService questService, + GameEngine gameEngine) { + this.gameRepository = gameRepository; + this.userService = userService; + this.questService = questService; + this.gameEngine = gameEngine; + } + + public Game startNewGame(Long questId, Long userId) { + User user = userService.findById(userId).orElseThrow(() -> new AppException(USER_NOT_FOUND)); + Quest quest = questService.findById(questId).orElseThrow(() -> new AppException(QUEST_NOT_FOUND)); + GameState initialState = gameEngine.startGame(user, quest); + Game game = Game.builder() + .questId(questId) + .userId(userId) + .currentQuestionId(initialState.getCurrentQuestion().getGeneratedId()) + .gameState(initialState) + .build(); + return gameRepository.save(game); + } + + public Game advanceGame(Long gameId, Long answerId) { + Game game = gameRepository.findById(gameId).orElseThrow(() -> new AppException("Game not found")); + GameState nextState = gameEngine.advanceGame(game.getGameState(), answerId); + game.setCurrentQuestionId(nextState.getCurrentQuestion().getGeneratedId()); + game.setGameState(nextState); + return gameRepository.save(game); + } + + public Optional getGameById(Long gameId) { + return gameRepository.findById(gameId); + } +} diff --git a/src/main/java/com/javarush/vasileva/service/QuestService.java b/src/main/java/com/javarush/vasileva/service/QuestService.java new file mode 100644 index 0000000..2b106fe --- /dev/null +++ b/src/main/java/com/javarush/vasileva/service/QuestService.java @@ -0,0 +1,51 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Quest; +import com.javarush.vasileva.repository.QuestRepository; +import com.javarush.vasileva.repository.Repository; +import com.javarush.vasileva.util.Helpers; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; +import java.util.Optional; + +public class QuestService { + + private static final Logger LOGGER = LoggerFactory.getLogger(QuestService.class.getName()); + + private final Repository questRepository; + + public QuestService(QuestRepository questRepository) { + this.questRepository = questRepository; + } + + public List getAll() { + return questRepository.getAll(); + } + + public Optional findById(Long questId) { + return questRepository.findById(questId); + } + + public void create(Quest quest) { + questRepository.create(quest); + } + + public void update(Quest quest) { + questRepository.update(quest); + } + + public void delete(Quest quest) { + questRepository.delete(quest); + } + + public Optional getValidatedQuest(String questIdStr) { + if (questIdStr == null || questIdStr.isEmpty()) { + LOGGER.warn("Missing quest ID in DELETE request"); + return Optional.empty(); + } + Long questId = Helpers.parseStringToLong(questIdStr); + return findById(questId); + } +} diff --git a/src/main/java/com/javarush/vasileva/service/QuestionService.java b/src/main/java/com/javarush/vasileva/service/QuestionService.java new file mode 100644 index 0000000..aa44766 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/service/QuestionService.java @@ -0,0 +1,38 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Answer; +import com.javarush.vasileva.entity.Question; +import com.javarush.vasileva.repository.QuestionRepository; + +import java.util.List; +import java.util.Optional; + +public class QuestionService { + + private final QuestionRepository questionRepository; + + public QuestionService(QuestionRepository questionRepository) { + this.questionRepository = questionRepository; + } + + public void create(Question question) { + questionRepository.create(question); + } + + public List getAll() { + return questionRepository.getAll(); + } + + public Optional findById(Long id) { + return questionRepository.findById(id); + } + + public Optional getByQuestionLabelAndQuestId(String questionLabel, long questId) { + return questionRepository.getByQuestionLabelAndQuestId(questionLabel, questId); + } + + public boolean isFinalQuestion(Question question) { + List answers = question.getAnswers(); + return answers == null || answers.isEmpty(); + } +} diff --git a/src/main/java/com/javarush/vasileva/service/UserService.java b/src/main/java/com/javarush/vasileva/service/UserService.java new file mode 100644 index 0000000..43d3fa0 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/service/UserService.java @@ -0,0 +1,61 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Role; +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.repository.UserRepository; +import com.javarush.vasileva.util.Helpers; + +import java.util.List; +import java.util.Optional; + +public class UserService { + + private final UserRepository userRepository; + + public UserService(UserRepository userRepository) { + this.userRepository = userRepository; + } + + public void create(User user) { + userRepository.create(user); + } + + public void update(User user) { + userRepository.update(user); + } + + public void delete(User user) { + userRepository.delete(user); + } + + public List getAll() { + return userRepository.getAll(); + } + + public Optional findById(Long userId) { + return userRepository.findById(userId); + } + + public void register(String login, String email, String password) { + User user = User.builder() + .login(login) + .email(email) + .password(password) + .role(Role.USER) + .build(); + userRepository.create(user); + } + + public Optional login(String email, String password) { + return userRepository.findByEmail(email) + .filter(user -> user.getPassword().equals(password)); + } + + public Optional getValidatedUser(String userIdStr) { + if (userIdStr == null || userIdStr.isEmpty()) { + return Optional.empty(); + } + Long questId = Helpers.parseStringToLong(userIdStr); + return findById(questId); + } +} diff --git a/src/main/java/com/javarush/vasileva/service/UserStatsService.java b/src/main/java/com/javarush/vasileva/service/UserStatsService.java new file mode 100644 index 0000000..558797b --- /dev/null +++ b/src/main/java/com/javarush/vasileva/service/UserStatsService.java @@ -0,0 +1,37 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Question; +import com.javarush.vasileva.entity.UserStats; +import com.javarush.vasileva.repository.UserStatsRepository; + +import java.util.Optional; + +import static com.javarush.vasileva.util.Value.LOSS; +import static com.javarush.vasileva.util.Value.WIN; + +public class UserStatsService { + private final UserStatsRepository userStatsRepository; + + public UserStatsService(UserStatsRepository userStatsRepository) { + this.userStatsRepository = userStatsRepository; + } + + public UserStats createUserStats(long userId) { + return userStatsRepository.createUserStats(userId); + } + + public Optional getUserStats(long userId) { + return userStatsRepository.getUserStats(userId); + } + + public void updateUserStats(Question question, UserStats userStats) { + userStats.setTotal(userStats.getTotal() + 1); + if (question.getLabel().contains(WIN)) { + userStats.setWins(userStats.getWins() + 1); + } else if (question.getLabel().contains(LOSS)) { + userStats.setLosses(userStats.getLosses() + 1); + } + userStatsRepository.updateUserStats(userStats); + } + +} diff --git a/src/main/java/com/javarush/vasileva/util/Helpers.java b/src/main/java/com/javarush/vasileva/util/Helpers.java new file mode 100644 index 0000000..b2768de --- /dev/null +++ b/src/main/java/com/javarush/vasileva/util/Helpers.java @@ -0,0 +1,15 @@ +package com.javarush.vasileva.util; + +public class Helpers { + + private Helpers() { + } + + public static Long parseStringToLong(String str) { + try { + return Long.parseLong(str); + } catch (NumberFormatException e) { + throw new IllegalArgumentException("Invalid quest id"); + } + } +} diff --git a/src/main/java/com/javarush/vasileva/util/Key.java b/src/main/java/com/javarush/vasileva/util/Key.java new file mode 100644 index 0000000..a9fdf99 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/util/Key.java @@ -0,0 +1,30 @@ +package com.javarush.vasileva.util; + +public class Key { + public static final String USER_ID = "id"; + public static final String USER = "user"; + public static final String USERS = "users"; + public static final String LOGIN = "login"; + public static final String EMAIL = "email"; + public static final String PASSWORD = "password"; + public static final String ROLE = "role"; + public static final String QUESTS = "quests"; + public static final String QUEST_ID = "questId"; + public static final String QUEST = "quest"; + public static final String GAME = "game"; + public static final String GAME_ID = "gameId"; + public static final String EDIT = "edit"; + public static final String QUEST_JSON = "questJson"; + public static final String SELECTED_ANSWER_ID = "selectedAnswerId"; + public static final String STATE = "state"; + public static final String STATS = "stats"; + public static final String WINNING = "winning"; + public static final String CREATE = "create"; + public static final String UPDATE = "update"; + + public static final String ERROR = "error"; + + public static final String QUEST_FILE_NAME_1 = "/json/quest-1.json"; + public static final String QUEST_FILE_NAME_2 = "/json/quest-2.json"; + public static final String QUEST_FILE_NAME_3 = "/json/quest-3.json"; +} diff --git a/src/main/java/com/javarush/vasileva/util/Link.java b/src/main/java/com/javarush/vasileva/util/Link.java new file mode 100644 index 0000000..538f7f3 --- /dev/null +++ b/src/main/java/com/javarush/vasileva/util/Link.java @@ -0,0 +1,17 @@ +package com.javarush.vasileva.util; + +public class Link { + public static final String INDEX = ""; + public static final String HOME = "/home"; + public static final String LOGIN = "/login"; + public static final String LOGOUT = "/logout"; + public static final String REGISTER = "/register"; + public static final String PROFILE = "/profile"; + public static final String USER_LIST = "/user-list"; + public static final String EDIT_USER = "/edit-user"; + public static final String PLAY_GAME = "/play-game"; + public static final String EDIT_QUEST = "/edit-quest"; + public static final String STATS = "/stats"; + public static final String ERROR = "/error-page"; + +} diff --git a/src/main/java/com/javarush/vasileva/util/Value.java b/src/main/java/com/javarush/vasileva/util/Value.java new file mode 100644 index 0000000..7960c8f --- /dev/null +++ b/src/main/java/com/javarush/vasileva/util/Value.java @@ -0,0 +1,52 @@ +package com.javarush.vasileva.util; + +public class Value { + public static final String JSON_SAMPLE = """ + { + "title": "Название", + "description": "Краткое описание", + "text": "Текст", + "image": "background-main.png", + "questions": [ + { + "label": "1", + "text": "Вопрос 1", + "answers": [ + { + "nextQuestionLabel": "+810", + "text": "Вариант ответа 1" + }, + { + "nextQuestionLabel": "-910", + "text": "Вариант ответа 2" + } + ] + }, + { + "label": "+810", + "text": "Победа!" + }, + { + "label": "-910", + "text": "Поражение..." + } + ] + }"""; + + public static final String WIN = "+"; + public static final String LOSS = "-"; + + public static final String JSON_SAVE_ERROR = "Ошибка при сохранении квеста"; + public static final String QUEST_NOT_FOUND = "Квест не найден: id="; + public static final String QUESTION_NOT_FOUND = "Вопрос не найден: label="; + public static final String GAME_NOT_FOUND = "Игра не найдена"; + public static final String ANSWER_NOT_FOUND = "Ответ не найден"; + public static final String STATS_NOT_FOUND = "Статистика не найдена"; + public static final String EDIT_QUEST_AUTH_ERROR = "Редактировать квесты могут только пользователи с правами ADMIN"; + public static final String DELETE_QUEST_AUTH_ERROR = "Удалять квесты могут только пользователи с правами ADMIN"; + public static final String USER_LIST_AUTH_ERROR = "Получить список пользователей могут только пользователи с правами ADMIN"; + public static final String USER_NOT_FOUND = "Пользователь не найден: id="; + public static final String EMPTY_DATA_ERROR = "Необходимо ввести данные"; + public static final String INVALID_DATA_ERROR = "Неверный email или пароль"; + public static final String AUTH_ERROR = "Необходимо авторизоваться"; +} diff --git a/src/main/resources/json/quest-1.json b/src/main/resources/json/quest-1.json new file mode 100644 index 0000000..1526205 --- /dev/null +++ b/src/main/resources/json/quest-1.json @@ -0,0 +1,62 @@ +{ + "title": "Космическое приключение", + "description": "Ты — космический путешественник, чей корабль внезапно получает загадочный сигнал от неопознанного объекта...", + "text": "Ты — космический путешественник, чей корабль внезапно получает загадочный сигнал от неопознанного объекта на границе изученной галактики. Впереди — неизвестность, за каждым поворотом таятся тайны космоса, а каждое решение может стать судьбоносным. Готов ли ты принять вызов Вселенной и отправиться в путешествие, где правда и ложь, смелость и осторожность, судьба и случай переплетутся в одну невероятную историю?", + "image": "quest-1.png", + "questions": [ + { + "label": "1", + "text": "Принять вызов НЛО?", + "answers": [ + { + "nextQuestionLabel": "2", + "text": "Принять вызов" + }, + { + "nextQuestionLabel": "-910", + "text": "Отклонить вызов" + } + ] + }, + { + "label": "2", + "text": "Поднимешься на мостик к капитану?", + "answers": [ + { + "nextQuestionLabel": "3", + "text": "Подняться на мостик" + }, + { + "nextQuestionLabel": "-910", + "text": "Отказаться подниматься на мостик" + } + ] + }, + { + "label": "3", + "text": "Ты поднялся на мостик. Ты кто?", + "answers": [ + { + "nextQuestionLabel": "+810", + "text": "Рассказать правду о себе" + }, + { + "nextQuestionLabel": "-911", + "text": "Солгать о себе" + } + ] + }, + { + "label": "+810", + "text": "Победа! Тебя вернули домой." + }, + { + "label": "-910", + "text": "Поражение... Ты отклонил вызов" + }, + { + "label": "-911", + "text": "Поражение... Твою ложь разоблачили" + } + ] +} diff --git a/src/main/resources/json/quest-2.json b/src/main/resources/json/quest-2.json new file mode 100644 index 0000000..6e3b1cc --- /dev/null +++ b/src/main/resources/json/quest-2.json @@ -0,0 +1,457 @@ +{ + "title": "Автостопом по здравому смыслу (или Не паникуй)", + "description": "По мотивам «Автостопом по Галактике»", + "text": "Добро пожаловать в квест, который начинается с конца света и быстро переходит к вопросам посложнее. Землю сносят ради гиперпространственной магистрали, предупреждения были вывешены (где-то далеко и давно), а у вас есть лишь выбор, полотенце и крайне смутное представление о происходящем. Впереди — вогоны, сомнительный чай, философские размышления и шанс узнать ответ на Главный вопрос жизни, Вселенной и всего такого. Главное — не паникуйте: Вселенная всё равно делает вид, что всё под контролем.", + "image": "quest-2.png", + "questions": [ + { + "label": "1", + "text": "Вы просыпаетесь и узнаёте, что Землю демонтируют ради гиперпространственной магистрали. Что будете делать?", + "answers": [ + { + "nextQuestionLabel": "2", + "text": "Вы возмущённо требуете документы, разрешения и хотя бы подпись ответственного лица. Вселенная равнодушно зевает." + }, + { + "nextQuestionLabel": "3", + "text": "Вы вспоминаете главное правило галактического путешественника и лихорадочно ищете полотенце. Это внушает необъяснимое спокойствие." + }, + { + "nextQuestionLabel": "4", + "text": "Вы уточняете, будет ли обед до конца света, потому что на голодный желудок апокалипсис переносится хуже." + }, + { + "nextQuestionLabel": "5", + "text": "Вы решаете, что это сон, и пробуете снова уснуть, пока Вселенная не передумала." + } + ] + }, + { + "label": "2", + "text": "Перед вами стоит вогон, держащий папку подозрительно толстых стихов.", + "answers": [ + { + "nextQuestionLabel": "6", + "text": "Вы пытаетесь убежать, но быстро понимаете, что в условиях вакуума это не лучшая стратегия." + }, + { + "nextQuestionLabel": "7", + "text": "Вы представляете себя галактическим юристом и начинаете говорить словами, смысл которых неизвестен даже вам." + }, + { + "nextQuestionLabel": "8", + "text": "Вы вежливо просите почитать стихи вслух, подозревая, что хуже уже не будет." + }, + { + "nextQuestionLabel": "9", + "text": "Вы начинаете плакать заранее, экономя время на эмоциональной реакции." + } + ] + }, + { + "label": "3", + "text": "Полотенце найдено. Оно слегка влажное, но внушает доверие.", + "answers": [ + { + "nextQuestionLabel": "6", + "text": "Вы используете его как универсальное средство защиты от всего непонятного." + }, + { + "nextQuestionLabel": "10", + "text": "Вы накидываете его как плащ и чувствуете себя героем космооперы." + }, + { + "nextQuestionLabel": "5", + "text": "Вы обнимаете полотенце, потому что оно единственное, что сейчас вас понимает." + }, + { + "nextQuestionLabel": "9", + "text": "Вы тут же его теряете, подтверждая законы Вселенной." + } + ] + }, + { + "label": "4", + "text": "Мысли о еде не покидают вас даже в момент уничтожения планеты.", + "answers": [ + { + "nextQuestionLabel": "11", + "text": "Вы решаете сделать бутерброд, игнорируя катастрофу как нечто второстепенное." + }, + { + "nextQuestionLabel": "12", + "text": "Вы жалуетесь Вселенной на несвоевременность происходящего." + }, + { + "nextQuestionLabel": "5", + "text": "Вы смиряетесь с судьбой и принимаете происходящее как должное." + }, + { + "nextQuestionLabel": "13", + "text": "Вы спрашиваете, где чай, потому что без него цивилизация не имеет смысла." + } + ] + }, + { + "label": "5", + "text": "Вы понимаете: это не сон, а реальность, причём весьма навязчивая.", + "answers": [ + { + "nextQuestionLabel": "9", + "text": "Вы начинаете паниковать, громко и без системы." + }, + { + "nextQuestionLabel": "10", + "text": "Вы вспоминаете надпись «DON’T PANIC» и стараетесь выглядеть уверенно." + }, + { + "nextQuestionLabel": "12", + "text": "Вы философски принимаете происходящее, как будто читали об этом раньше.", + "description": "" + }, + { + "nextQuestionLabel": "14", + "text": "Вы выкрикиваете «42!» без всякой причины, но с чувством." + } + ] + }, + { + "label": "6", + "text": "Вас без особых церемоний выбрасывают в открытый космос. Он холодный, бесконечный и явно не собирается извиняться.", + "answers": [ + { + "nextQuestionLabel": "15", + "text": "Вы задерживаете дыхание, словно космос обязан это заметить и проявить такт." + }, + { + "nextQuestionLabel": "9", + "text": "Вы начинаете кричать, несмотря на смутное подозрение, что звук здесь не работает." + }, + { + "nextQuestionLabel": "16", + "text": "Вы решаете положиться на сюжетную броню главного героя, даже если не уверены, что вы главный." + }, + { + "nextQuestionLabel": "12", + "text": "Вы вежливо извиняетесь перед космосом за возможные неудобства. Космос впечатлён." + } + ] + }, + { + "label": "7", + "text": "Вы представились галактическим юристом, и теперь все ждут продолжения.", + "answers": [ + { + "nextQuestionLabel": "8", + "text": "Вы требуете апелляцию, не уточняя, к чему именно. Звучит убедительно." + }, + { + "nextQuestionLabel": "9", + "text": "Вы начинаете путаться в терминах и случайно признаётесь во всех преступлениях сразу." + }, + { + "nextQuestionLabel": "10", + "text": "Против всех законов логики вы выигрываете дело. Вогон выглядит озадаченным." + }, + { + "nextQuestionLabel": "8", + "text": "В качестве «компромисса» вам предлагают послушать стихи." + } + ] + }, + { + "label": "8", + "text": "Поэзия начинается. Это заметно сразу.", + "answers": [ + { + "nextQuestionLabel": "15", + "text": "Вы теряете сознание, руководствуясь инстинктом самосохранения." + }, + { + "nextQuestionLabel": "6", + "text": "Вы хвалите рифму, чем пугаете всех присутствующих, включая автора." + }, + { + "nextQuestionLabel": "9", + "text": "Вы записываете цитаты, чтобы пугать ими знакомых.", + "description": "Не хватает длины." + }, + { + "nextQuestionLabel": "15", + "text": "Вы просите ещё. Реальность слегка трескается." + } + ] + }, + { + "label": "9", + "text": "Ситуация становится заметно хуже, чем была мгновение назад.", + "answers": [ + { + "nextQuestionLabel": "5", + "text": "Вы решаете сдаться и начать всё сначала." + }, + { + "nextQuestionLabel": "12", + "text": "Вы делаете вид, что так и было задумано." + }, + { + "nextQuestionLabel": "14", + "text": "Вы задаёте главный философский вопрос: «В чём вообще смысл?»" + }, + { + "nextQuestionLabel": "3", + "text": "Вы снова начинаете искать полотенце, потому что оно вас ещё ни разу не подводило." + } + ] + }, + { + "label": "10", + "text": "Вас подбирает корабль, который выглядит надёжным ровно настолько, насколько это возможно в галактике.", + "answers": [ + { + "nextQuestionLabel": "16", + "text": "Вы радуетесь спасению и решаете не задавать лишних вопросов." + }, + { + "nextQuestionLabel": "17", + "text": "Вы внимательно разглядываете экипаж, подозревая худшее." + }, + { + "nextQuestionLabel": "9", + "text": "Вы прячетесь за ближайшим ящиком. На всякий случай." + }, + { + "nextQuestionLabel": "11", + "text": "Вы спрашиваете, куда летит корабль, и сразу жалеете об этом." + } + ] + }, + { + "label": "11", + "text": "На борту обсуждают еду, и это внезапно кажется важным.", + "answers": [ + { + "nextQuestionLabel": "13", + "text": "Вы заказываете чай, надеясь, что в этот раз всё будет иначе." + }, + { + "nextQuestionLabel": "14", + "text": "Вы снова поднимаете тему смысла жизни. Не вовремя, но искренне." + }, + { + "nextQuestionLabel": "9", + "text": "Вы пробуете что-то странное, светящееся и, возможно, разумное." + }, + { + "nextQuestionLabel": "12", + "text": "Вы решаете ничего не есть и просто подумать." + } + ] + }, + { + "label": "12", + "text": "Вы внезапно ловите себя на том, что задумались. Это опасно, но уже поздно.", + "answers": [ + { + "nextQuestionLabel": "14", + "text": "Вы размышляете о жизни: зачем она дана, почему всё так странно и кто вообще всё это придумал." + }, + { + "nextQuestionLabel": "14", + "text": "Вы думаете о Вселенной — бесконечной, равнодушной и удивительно хорошо организованной для хаоса." + }, + { + "nextQuestionLabel": "14", + "text": "Вы пытаетесь осмыслить всё сразу: жизнь, Вселенную и остальное, но мысль ускользает." + }, + { + "nextQuestionLabel": "5", + "text": "Вы решаете, что думать — занятие переоценённое, и возвращаетесь к реальности." + } + ] + }, + { + "label": "13", + "text": "Вам приносят чай. Он снова не такой, как надо.", + "answers": [ + { + "nextQuestionLabel": "9", + "text": "Вы возмущаетесь, потому что чай — это вопрос принципа, а не вкуса." + }, + { + "nextQuestionLabel": "17", + "text": "Вы пытаетесь улучшить напиток научным методом, добавляя ингредиенты сомнительного происхождения." + }, + { + "nextQuestionLabel": "12", + "text": "Вы философски принимаете чай как символ несовершенства Вселенной." + }, + { + "nextQuestionLabel": "3", + "text": "Вы отвлекаетесь и спрашиваете, где ваше полотенце." + } + ] + }, + { + "label": "14", + "text": "Наступает торжественный момент. Вам задают Главный вопрос жизни, Вселенной и всего такого. Все замолкают. Даже космос делает вид, что слушает.", + "answers": [ + { + "nextQuestionLabel": "9", + "text": "Вы отвечаете: «Любовь». Это звучит красиво, но Вселенная выглядит разочарованной." + }, + { + "nextQuestionLabel": "9", + "text": "Вы честно говорите: «Деньги». Вселенная вздыхает и делает пометку." + }, + { + "nextQuestionLabel": "20", + "text": "Вы уверенно и без лишних слов произносите: 42. Вселенная одобрительно молчит. Это хороший знак." + }, + { + "nextQuestionLabel": "9", + "text": "Вы заявляете, что вопрос сформулирован некорректно. Вселенная кивает, но продолжает эксперимент." + } + ] + }, + { + "label": "15", + "text": "Вы обнаруживаете, что каким-то образом всё ещё существуете.", + "answers": [ + { + "nextQuestionLabel": "16", + "text": "Вы благодарите Вселенную за то, что она сегодня в хорошем настроении." + }, + { + "nextQuestionLabel": "12", + "text": "Вы долго и искренне удивляетесь происходящему." + }, + { + "nextQuestionLabel": "5", + "text": "Вы проверяете, действительно ли вы живы, на всякий случай." + }, + { + "nextQuestionLabel": "17", + "text": "Вы решаете, что это странно, но, вероятно, нормально." + } + ] + }, + { + "label": "16", + "text": "У вас появляется ощущение, что вы близки к истине. Это тревожит.", + "answers": [ + { + "nextQuestionLabel": "20", + "text": "Вы решаете продолжить путешествие, не вдаваясь в детали." + }, + { + "nextQuestionLabel": "14", + "text": "Вы задаёте ещё один вопрос, потому что не умеете останавливаться." + }, + { + "nextQuestionLabel": "12", + "text": "Вы расслабляетесь и позволяете событиям происходить." + }, + { + "nextQuestionLabel": "3", + "text": "Вы автоматически проверяете, при вас ли полотенце." + } + ] + }, + { + "label": "17", + "text": "Перед вами андроид с крайне пессимистичным взглядом на бытие.", + "answers": [ + { + "nextQuestionLabel": "12", + "text": "Вы искренне сочувствуете ему. Андроид растерян: такого он не ожидал." + }, + { + "nextQuestionLabel": "9", + "text": "Вы позволяете ему высказаться. Через некоторое время жалеете об этом." + }, + { + "nextQuestionLabel": "14", + "text": "Вы спрашиваете его о смысле жизни. Он подтверждает, что всё плохо." + }, + { + "nextQuestionLabel": "16", + "text": "Вы медленно отходите, стараясь не привлекать внимания." + } + ] + }, + { + "label": "18", + "text": "Вселенная смотрит на вас с плохо скрываемым интересом.", + "answers": [ + { + "nextQuestionLabel": "12", + "text": "Вы почтительно кланяетесь, считая это уместным." + }, + { + "nextQuestionLabel": "16", + "text": "Вы улыбаетесь, демонстрируя уверенность, которой у вас нет." + }, + { + "nextQuestionLabel": "20", + "text": "Вы прямо спрашиваете: «Ну и что дальше?»" + }, + { + "nextQuestionLabel": "9", + "text": "Вы решаете убежать, пока всё снова не усложнилось." + } + ] + }, + { + "label": "19", + "text": "Вы чувствуете, что ответ уже близко, но всё ещё сомневаетесь.", + "answers": [ + { + "nextQuestionLabel": "20", + "text": "Вы делаете вывод и принимаете его без проверки.", + "description": "" + }, + { + "nextQuestionLabel": "9", + "text": "Вы решаете всё усложнить, потому что слишком просто не бывает.", + "description": "" + }, + { + "nextQuestionLabel": "14", + "text": "Вы спрашиваете ещё раз, на всякий случай." + }, + { + "nextQuestionLabel": "16", + "text": "Вы напоминаете себе главное правило — не паниковать." + } + ] + }, + { + "label": "20", + "text": "После всего пережитого вы наконец понимаете главное.", + "answers": [ + { + "nextQuestionLabel": "+810", + "text": "Всегда носи полотенце" + }, + { + "nextQuestionLabel": "+810", + "text": "Не паникуй" + }, + { + "nextQuestionLabel": "+810", + "text": "Ответ — 42" + }, + { + "nextQuestionLabel": "+810", + "text": "Всё вышеперечисленное" + } + ] + }, + { + "label": "+810", + "text": "Поздравляем! Вы прошли квест, не сломали Вселенную и, возможно, даже стали немного мудрее.", + "answers": null + } + ] +} diff --git a/src/main/resources/json/quest-3.json b/src/main/resources/json/quest-3.json new file mode 100644 index 0000000..af016e3 --- /dev/null +++ b/src/main/resources/json/quest-3.json @@ -0,0 +1,168 @@ +{ + "title": "Знаешь ли ты английский достаточно для того, чтобы играть в игры?", + "description": "Проверь свой базовый игровой словарный запас и понимание типичных фраз из англоязычных видеоигр", + "text": "Ты собираешься запустить новую RPG на английском, но не уверен, поймёшь ли диалоги и интерфейс. Давай проверим! На каждый вопрос — 3–4 варианта. Выбери один. Ошибёшься — игра «завершится».", + "image": "quest-3.png", + "questions": [ + { + "label": "1", + "text": "Что означает кнопка «Continue» в меню игры?", + "answers": [ + { + "nextQuestionLabel": "-910", + "text": "Начать заново" + }, + { + "nextQuestionLabel": "-910", + "text": "Сохранить игру" + }, + { + "nextQuestionLabel": "2", + "text": "Продолжить" + }, + { + "nextQuestionLabel": "-910", + "text": "Выйти" + } + ] + }, + { + "label": "2", + "text": "Как перевести «Quest accepted»?", + "answers": [ + { + "nextQuestionLabel": "-911", + "text": "Квест завершён" + }, + { + "nextQuestionLabel": "3", + "text": "Квест принят" + }, + { + "nextQuestionLabel": "-911", + "text": "Квест провалился»" + }, + { + "nextQuestionLabel": "-911", + "text": "Квест недоступен" + } + ] + }, + { + "label": "3", + "text": "Что значит «HP at critical level!»?", + "answers": [ + { + "nextQuestionLabel": "4", + "text": "Здоровье на критическом уровне!" + }, + { + "nextQuestionLabel": "-912", + "text": "Мана на критическом уровне!" + }, + { + "nextQuestionLabel": "-912", + "text": "Энергия на нуле!" + }, + { + "nextQuestionLabel": "-912", + "text": "Время истекает!" + } + ] + }, + { + "label": "4", + "text": "Какой вариант означает «Сохранить игру»?", + "answers": [ + { + "nextQuestionLabel": "-913", + "text": "Load game" + }, + { + "nextQuestionLabel": "5", + "text": "Save game" + }, + { + "nextQuestionLabel": "-913", + "text": "Delete game" + }, + { + "nextQuestionLabel": "-913", + "text": "Start game" + } + ] + }, + { + "label": "5", + "text": "Ты видишь диалог: «I’ll meet you at the crossroads. Be wary of bandits.» Что это значит?", + "answers": [ + { + "nextQuestionLabel": "6", + "text": "Я встречу тебя у перекрёстка. Остерегайся бандитов.»" + }, + { + "nextQuestionLabel": "-914", + "text": "Я жду тебя у реки. Будь осторожен с волками." + }, + { + "nextQuestionLabel": "-914", + "text": "Встретимся в таверне. Не доверяй незнакомцам." + }, + { + "nextQuestionLabel": "-914", + "text": "Иди на север. Там безопасно." + } + ] + }, + { + "label": "6", + "text": "В инвентаре написано: «Rusted sword (Durability: 20/100)». Что это значит?", + "answers": [ + { + "nextQuestionLabel": "-915", + "text": "Старый меч (Урон: 20/100)" + }, + { + "nextQuestionLabel": "-915", + "text": "Сломанный меч (Вес: 20/100)" + }, + { + "nextQuestionLabel": "+810", + "text": "Ржавый меч (Прочность: 20/100)" + }, + { + "nextQuestionLabel": "-915", + "text": "Меч с руной (Заряды: 20/100)" + } + ] + }, + { + "label": "+810", + "text": "Ты прошёл квест! Твой английский достаточно хорош, чтобы разбираться в интерфейсе, квестах и диалогах англоязычных игр." + }, + { + "label": "-910", + "text": "«Continue» — это «продолжить», а не «выйти» или «начать заново». Игра завершена. Попробуй ещё!" + }, + { + "label": "-911", + "text": "«Quest accepted» значит «Квест принят». Без понимания квестов далеко не уйдёшь. Игра завершена." + }, + { + "label": "-912", + "text": "«HP» = Health Points («здоровье»). Критический уровень HP — пора лечиться. Игра завершена." + }, + { + "label": "-913", + "text": "«Save game» — «Сохранить игру». «Load» — загрузить, «Delete» — удалить. Игра завершена." + }, + { + "label": "-914", + "text": "Правильный перевод: «Я встречу тебя у перекрёстка. Остерегайся бандитов». Без понимания диалогов не пройти сюжет. Игра завершена." + }, + { + "label": "-915", + "text": "«Durability» = «прочность». Без понимания параметров инвентаря сложно выжить. Игра завершена." + } + ] +} diff --git a/src/main/resources/log4j2.xml b/src/main/resources/log4j2.xml new file mode 100644 index 0000000..b261a23 --- /dev/null +++ b/src/main/resources/log4j2.xml @@ -0,0 +1,31 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/webapp/WEB-INF/edit-quest.jsp b/src/main/webapp/WEB-INF/edit-quest.jsp new file mode 100644 index 0000000..8321ed5 --- /dev/null +++ b/src/main/webapp/WEB-INF/edit-quest.jsp @@ -0,0 +1,64 @@ +<%@ page contentType="text/html;charset=UTF-8" isELIgnored="false" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> +<%@include file="header.jsp" %> + + + + + + + + ${requestScope.quest != null ? 'Редактирование квеста' : 'Создание квеста'} + + +
+
+
+
+
+
+ +

+ ${requestScope.edit ? 'Редактирование квеста' : 'Новый квест'} +

+ +

Для редактирования нажмите на карточку квеста "Редактировать квест" или выберите квест из + списка ниже

+ +
+ + + +
+ +
+
+ + +
+ + + + +
${sessionScope.error}
+
+
+
+
+
+
+
+
+ +<%@include file="footer.jsp" %> + + + + diff --git a/src/main/webapp/WEB-INF/edit-user.jsp b/src/main/webapp/WEB-INF/edit-user.jsp index f274104..5edcc32 100644 --- a/src/main/webapp/WEB-INF/edit-user.jsp +++ b/src/main/webapp/WEB-INF/edit-user.jsp @@ -1,48 +1,65 @@ -<%@ page contentType="text/html;charset=UTF-8" language="java" %> -<%@include file="head.jsp" %> +<%@ page contentType="text/html;charset=UTF-8" isELIgnored="false" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> +<%@include file="header.jsp" %> + + + + + -
+
- - Edit user: +

Добавление/редактирование пользователя

-
- +
+
+
+ +
+ +
+ - min 3 symbols + required>
-
- +
- min 8 symb + required + maxlength="15" + minlength="4">
- -
- +
+ + + +
+ +
+ +
+ +
+ + + diff --git a/src/main/webapp/WEB-INF/footer.jsp b/src/main/webapp/WEB-INF/footer.jsp new file mode 100644 index 0000000..286a6d3 --- /dev/null +++ b/src/main/webapp/WEB-INF/footer.jsp @@ -0,0 +1,11 @@ +<%@ page contentType="text/html;charset=UTF-8" %> + + + Title + + +
+

@KatiaVasileva, 2026.

+
+ + diff --git a/src/main/webapp/WEB-INF/head.jsp b/src/main/webapp/WEB-INF/head.jsp deleted file mode 100644 index 2f7b9f2..0000000 --- a/src/main/webapp/WEB-INF/head.jsp +++ /dev/null @@ -1,11 +0,0 @@ -<%@ page contentType="text/html;charset=UTF-8" language="java" %> -<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> - - - Title - - - \ No newline at end of file diff --git a/src/main/webapp/WEB-INF/header.jsp b/src/main/webapp/WEB-INF/header.jsp new file mode 100644 index 0000000..a4ee36b --- /dev/null +++ b/src/main/webapp/WEB-INF/header.jsp @@ -0,0 +1,67 @@ +<%@ page contentType="text/html;charset=UTF-8" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> + + + + Title + + + + + diff --git a/src/main/webapp/WEB-INF/home.jsp b/src/main/webapp/WEB-INF/home.jsp new file mode 100644 index 0000000..72c92a5 --- /dev/null +++ b/src/main/webapp/WEB-INF/home.jsp @@ -0,0 +1,79 @@ +<%@ page contentType="text/html;charset=UTF-8" isELIgnored="false" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> +<%@include file="header.jsp" %> + + + + + + + + Home + + +
+
+
+
+
+
+ background +
+
+
+ +
+
+ +
+

${quest.title}

+

${quest.description}

+
+ +
+
+ + +
+
+ + +
+
+ + +
+
+ +
+
+ + + +
+
+
+
+
+
+
+
+
+
+
+
+
+ +<%@include file="footer.jsp" %> + + + + + diff --git a/src/main/webapp/WEB-INF/list-user.jsp b/src/main/webapp/WEB-INF/list-user.jsp deleted file mode 100644 index dd52c55..0000000 --- a/src/main/webapp/WEB-INF/list-user.jsp +++ /dev/null @@ -1,9 +0,0 @@ -<%@ page contentType="text/html;charset=UTF-8" language="java" %> -<%@include file="head.jsp"%> - - - ${user.login} - - - - diff --git a/src/main/webapp/WEB-INF/login.jsp b/src/main/webapp/WEB-INF/login.jsp new file mode 100644 index 0000000..c4c5e03 --- /dev/null +++ b/src/main/webapp/WEB-INF/login.jsp @@ -0,0 +1,53 @@ +<%@ page contentType="text/html;charset=UTF-8" isELIgnored="false" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> +<%@include file="header.jsp" %> + + + + + + + + Title + + +
+
+
+

Вход в систему

+
+
+
+
+ + +
+
+ + +
+ +
+ +
+ +
+
+ +
${sessionScope.error}
+
+
+ + diff --git a/src/main/webapp/WEB-INF/play-game.jsp b/src/main/webapp/WEB-INF/play-game.jsp new file mode 100644 index 0000000..31d4b08 --- /dev/null +++ b/src/main/webapp/WEB-INF/play-game.jsp @@ -0,0 +1,82 @@ +<%@ page contentType="text/html;charset=UTF-8" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn" %> +<%@include file="header.jsp" %> + + + + + + + ${requestScope.quest.title} + + +
+
+
+ space +
+
+

${requestScope.quest.title}

+ +

${requestScope.quest.text}

+ + + +
+

${requestScope.game.gameState.currentQuestion.text}

+ Главная + Начать заново +
+
+ + +
+ + + +
+ Текущий вопрос + +
+ +
+ +
+ + +
+
+
+
+ + +
+
+
+
+ + +
${sessionScope.error}
+ <% session.removeAttribute("error"); %> +
+
+
+
+ +<%@include file="footer.jsp" %> + + diff --git a/src/main/webapp/WEB-INF/profile.jsp b/src/main/webapp/WEB-INF/profile.jsp new file mode 100644 index 0000000..a6be04f --- /dev/null +++ b/src/main/webapp/WEB-INF/profile.jsp @@ -0,0 +1,45 @@ +<%@ page contentType="text/html;charset=UTF-8" isELIgnored="false" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> +<%@include file="header.jsp" %> + + + + + + + + Profile + + + +
+
+
+

Ваш профиль

+
+
+
+
ID
+
${sessionScope.user.id}
+ +
Имя
+
${sessionScope.user.login}
+ +
Email
+
${sessionScope.user.email}
+ +
Роль
+
${sessionScope.user.role}
+
+ + +
+
+
+> + + diff --git a/src/main/webapp/WEB-INF/register.jsp b/src/main/webapp/WEB-INF/register.jsp new file mode 100644 index 0000000..074517d --- /dev/null +++ b/src/main/webapp/WEB-INF/register.jsp @@ -0,0 +1,63 @@ +<%@ page contentType="text/html;charset=UTF-8" isELIgnored="false" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> +<%@include file="header.jsp" %> + + + + + + + + Register + + +
+
+
+

Регистрация

+
+
+ +
+
+ + +
+
+ + +
+
+ + +
+ +
+ +
+ +
+
+
+ + diff --git a/src/main/webapp/WEB-INF/start-page.jsp b/src/main/webapp/WEB-INF/start-page.jsp deleted file mode 100644 index 0531c1c..0000000 --- a/src/main/webapp/WEB-INF/start-page.jsp +++ /dev/null @@ -1,8 +0,0 @@ -<%@ page contentType="text/html;charset=UTF-8" language="java" %> -<%@include file="head.jsp"%> - -

<%= "Hello World!" %> -

-
-List Users - diff --git a/src/main/webapp/WEB-INF/stats.jsp b/src/main/webapp/WEB-INF/stats.jsp new file mode 100644 index 0000000..6c2d88c --- /dev/null +++ b/src/main/webapp/WEB-INF/stats.jsp @@ -0,0 +1,32 @@ + +<%@ page contentType="text/html;charset=UTF-8" isELIgnored="false" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> +<%@include file="header.jsp" %> + + + + + + + + Statistics + + +
+

Статистика игрока

+ + +
+
+
Игрок: ${sessionScope.user.login}
+

+ Пройдено квестов: ${sessionScope.stats.total}
+ Победы: ${sessionScope.stats.wins}
+ Поражения: ${sessionScope.stats.losses}
+

+
+
+
+
+ + diff --git a/src/main/webapp/WEB-INF/user-list.jsp b/src/main/webapp/WEB-INF/user-list.jsp new file mode 100644 index 0000000..b687644 --- /dev/null +++ b/src/main/webapp/WEB-INF/user-list.jsp @@ -0,0 +1,65 @@ +<%@ page contentType="text/html;charset=UTF-8" isELIgnored="false" %> +<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> +<%@include file="header.jsp" %> + + + + + + + + + + + + +
+

Список пользователей

+ + + + + + + + + + + + + + + + + + + + + + + +
IDИмяEmailРольДействия
${user.id}${user.login}${user.email}${user.role} + + Редактировать + + +
+ + + +
+
+ + + Добавить пользователя + +
+ + + diff --git a/src/main/webapp/images/background-main.png b/src/main/webapp/images/background-main.png new file mode 100644 index 0000000..e96fe70 Binary files /dev/null and b/src/main/webapp/images/background-main.png differ diff --git a/src/main/webapp/images/cat.png b/src/main/webapp/images/cat.png deleted file mode 100644 index 41771a1..0000000 Binary files a/src/main/webapp/images/cat.png and /dev/null differ diff --git a/src/main/webapp/images/quest-1.png b/src/main/webapp/images/quest-1.png new file mode 100644 index 0000000..662ece9 Binary files /dev/null and b/src/main/webapp/images/quest-1.png differ diff --git a/src/main/webapp/images/quest-2.png b/src/main/webapp/images/quest-2.png new file mode 100644 index 0000000..679be70 Binary files /dev/null and b/src/main/webapp/images/quest-2.png differ diff --git a/src/main/webapp/images/quest-3.png b/src/main/webapp/images/quest-3.png new file mode 100644 index 0000000..e240ebd Binary files /dev/null and b/src/main/webapp/images/quest-3.png differ diff --git a/src/main/webapp/static/bootstrap/css/bootstrap.min.css b/src/main/webapp/static/bootstrap/css/bootstrap.min.css new file mode 100644 index 0000000..da4b76b --- /dev/null +++ b/src/main/webapp/static/bootstrap/css/bootstrap.min.css @@ -0,0 +1,5 @@ +@charset "UTF-8";/*! + * Bootstrap v5.3.6 (https://getbootstrap.com/) + * Copyright 2011-2025 The Bootstrap Authors + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + */:root,[data-bs-theme=light]{--bs-blue:#0d6efd;--bs-indigo:#6610f2;--bs-purple:#6f42c1;--bs-pink:#d63384;--bs-red:#dc3545;--bs-orange:#fd7e14;--bs-yellow:#ffc107;--bs-green:#198754;--bs-teal:#20c997;--bs-cyan:#0dcaf0;--bs-black:#000;--bs-white:#fff;--bs-gray:#6c757d;--bs-gray-dark:#343a40;--bs-gray-100:#f8f9fa;--bs-gray-200:#e9ecef;--bs-gray-300:#dee2e6;--bs-gray-400:#ced4da;--bs-gray-500:#adb5bd;--bs-gray-600:#6c757d;--bs-gray-700:#495057;--bs-gray-800:#343a40;--bs-gray-900:#212529;--bs-primary:#0d6efd;--bs-secondary:#6c757d;--bs-success:#198754;--bs-info:#0dcaf0;--bs-warning:#ffc107;--bs-danger:#dc3545;--bs-light:#f8f9fa;--bs-dark:#212529;--bs-primary-rgb:13,110,253;--bs-secondary-rgb:108,117,125;--bs-success-rgb:25,135,84;--bs-info-rgb:13,202,240;--bs-warning-rgb:255,193,7;--bs-danger-rgb:220,53,69;--bs-light-rgb:248,249,250;--bs-dark-rgb:33,37,41;--bs-primary-text-emphasis:#052c65;--bs-secondary-text-emphasis:#2b2f32;--bs-success-text-emphasis:#0a3622;--bs-info-text-emphasis:#055160;--bs-warning-text-emphasis:#664d03;--bs-danger-text-emphasis:#58151c;--bs-light-text-emphasis:#495057;--bs-dark-text-emphasis:#495057;--bs-primary-bg-subtle:#cfe2ff;--bs-secondary-bg-subtle:#e2e3e5;--bs-success-bg-subtle:#d1e7dd;--bs-info-bg-subtle:#cff4fc;--bs-warning-bg-subtle:#fff3cd;--bs-danger-bg-subtle:#f8d7da;--bs-light-bg-subtle:#fcfcfd;--bs-dark-bg-subtle:#ced4da;--bs-primary-border-subtle:#9ec5fe;--bs-secondary-border-subtle:#c4c8cb;--bs-success-border-subtle:#a3cfbb;--bs-info-border-subtle:#9eeaf9;--bs-warning-border-subtle:#ffe69c;--bs-danger-border-subtle:#f1aeb5;--bs-light-border-subtle:#e9ecef;--bs-dark-border-subtle:#adb5bd;--bs-white-rgb:255,255,255;--bs-black-rgb:0,0,0;--bs-font-sans-serif:system-ui,-apple-system,"Segoe UI",Roboto,"Helvetica Neue","Noto Sans","Liberation Sans",Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--bs-font-monospace:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;--bs-gradient:linear-gradient(180deg, rgba(255, 255, 255, 0.15), rgba(255, 255, 255, 0));--bs-body-font-family:var(--bs-font-sans-serif);--bs-body-font-size:1rem;--bs-body-font-weight:400;--bs-body-line-height:1.5;--bs-body-color:#212529;--bs-body-color-rgb:33,37,41;--bs-body-bg:#fff;--bs-body-bg-rgb:255,255,255;--bs-emphasis-color:#000;--bs-emphasis-color-rgb:0,0,0;--bs-secondary-color:rgba(33, 37, 41, 0.75);--bs-secondary-color-rgb:33,37,41;--bs-secondary-bg:#e9ecef;--bs-secondary-bg-rgb:233,236,239;--bs-tertiary-color:rgba(33, 37, 41, 0.5);--bs-tertiary-color-rgb:33,37,41;--bs-tertiary-bg:#f8f9fa;--bs-tertiary-bg-rgb:248,249,250;--bs-heading-color:inherit;--bs-link-color:#0d6efd;--bs-link-color-rgb:13,110,253;--bs-link-decoration:underline;--bs-link-hover-color:#0a58ca;--bs-link-hover-color-rgb:10,88,202;--bs-code-color:#d63384;--bs-highlight-color:#212529;--bs-highlight-bg:#fff3cd;--bs-border-width:1px;--bs-border-style:solid;--bs-border-color:#dee2e6;--bs-border-color-translucent:rgba(0, 0, 0, 0.175);--bs-border-radius:0.375rem;--bs-border-radius-sm:0.25rem;--bs-border-radius-lg:0.5rem;--bs-border-radius-xl:1rem;--bs-border-radius-xxl:2rem;--bs-border-radius-2xl:var(--bs-border-radius-xxl);--bs-border-radius-pill:50rem;--bs-box-shadow:0 0.5rem 1rem rgba(0, 0, 0, 0.15);--bs-box-shadow-sm:0 0.125rem 0.25rem rgba(0, 0, 0, 0.075);--bs-box-shadow-lg:0 1rem 3rem rgba(0, 0, 0, 0.175);--bs-box-shadow-inset:inset 0 1px 2px rgba(0, 0, 0, 0.075);--bs-focus-ring-width:0.25rem;--bs-focus-ring-opacity:0.25;--bs-focus-ring-color:rgba(13, 110, 253, 0.25);--bs-form-valid-color:#198754;--bs-form-valid-border-color:#198754;--bs-form-invalid-color:#dc3545;--bs-form-invalid-border-color:#dc3545}[data-bs-theme=dark]{color-scheme:dark;--bs-body-color:#dee2e6;--bs-body-color-rgb:222,226,230;--bs-body-bg:#212529;--bs-body-bg-rgb:33,37,41;--bs-emphasis-color:#fff;--bs-emphasis-color-rgb:255,255,255;--bs-secondary-color:rgba(222, 226, 230, 0.75);--bs-secondary-color-rgb:222,226,230;--bs-secondary-bg:#343a40;--bs-secondary-bg-rgb:52,58,64;--bs-tertiary-color:rgba(222, 226, 230, 0.5);--bs-tertiary-color-rgb:222,226,230;--bs-tertiary-bg:#2b3035;--bs-tertiary-bg-rgb:43,48,53;--bs-primary-text-emphasis:#6ea8fe;--bs-secondary-text-emphasis:#a7acb1;--bs-success-text-emphasis:#75b798;--bs-info-text-emphasis:#6edff6;--bs-warning-text-emphasis:#ffda6a;--bs-danger-text-emphasis:#ea868f;--bs-light-text-emphasis:#f8f9fa;--bs-dark-text-emphasis:#dee2e6;--bs-primary-bg-subtle:#031633;--bs-secondary-bg-subtle:#161719;--bs-success-bg-subtle:#051b11;--bs-info-bg-subtle:#032830;--bs-warning-bg-subtle:#332701;--bs-danger-bg-subtle:#2c0b0e;--bs-light-bg-subtle:#343a40;--bs-dark-bg-subtle:#1a1d20;--bs-primary-border-subtle:#084298;--bs-secondary-border-subtle:#41464b;--bs-success-border-subtle:#0f5132;--bs-info-border-subtle:#087990;--bs-warning-border-subtle:#997404;--bs-danger-border-subtle:#842029;--bs-light-border-subtle:#495057;--bs-dark-border-subtle:#343a40;--bs-heading-color:inherit;--bs-link-color:#6ea8fe;--bs-link-hover-color:#8bb9fe;--bs-link-color-rgb:110,168,254;--bs-link-hover-color-rgb:139,185,254;--bs-code-color:#e685b5;--bs-highlight-color:#dee2e6;--bs-highlight-bg:#664d03;--bs-border-color:#495057;--bs-border-color-translucent:rgba(255, 255, 255, 0.15);--bs-form-valid-color:#75b798;--bs-form-valid-border-color:#75b798;--bs-form-invalid-color:#ea868f;--bs-form-invalid-border-color:#ea868f}*,::after,::before{box-sizing:border-box}@media (prefers-reduced-motion:no-preference){:root{scroll-behavior:smooth}}body{margin:0;font-family:var(--bs-body-font-family);font-size:var(--bs-body-font-size);font-weight:var(--bs-body-font-weight);line-height:var(--bs-body-line-height);color:var(--bs-body-color);text-align:var(--bs-body-text-align);background-color:var(--bs-body-bg);-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent}hr{margin:1rem 0;color:inherit;border:0;border-top:var(--bs-border-width) solid;opacity:.25}.h1,.h2,.h3,.h4,.h5,.h6,h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem;font-weight:500;line-height:1.2;color:var(--bs-heading-color)}.h1,h1{font-size:calc(1.375rem + 1.5vw)}@media (min-width:1200px){.h1,h1{font-size:2.5rem}}.h2,h2{font-size:calc(1.325rem + .9vw)}@media (min-width:1200px){.h2,h2{font-size:2rem}}.h3,h3{font-size:calc(1.3rem + .6vw)}@media (min-width:1200px){.h3,h3{font-size:1.75rem}}.h4,h4{font-size:calc(1.275rem + .3vw)}@media (min-width:1200px){.h4,h4{font-size:1.5rem}}.h5,h5{font-size:1.25rem}.h6,h6{font-size:1rem}p{margin-top:0;margin-bottom:1rem}abbr[title]{-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;-webkit-text-decoration-skip-ink:none;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}ol,ul{padding-left:2rem}dl,ol,ul{margin-top:0;margin-bottom:1rem}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem}b,strong{font-weight:bolder}.small,small{font-size:.875em}.mark,mark{padding:.1875em;color:var(--bs-highlight-color);background-color:var(--bs-highlight-bg)}sub,sup{position:relative;font-size:.75em;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:rgba(var(--bs-link-color-rgb),var(--bs-link-opacity,1));text-decoration:underline}a:hover{--bs-link-color-rgb:var(--bs-link-hover-color-rgb)}a:not([href]):not([class]),a:not([href]):not([class]):hover{color:inherit;text-decoration:none}code,kbd,pre,samp{font-family:var(--bs-font-monospace);font-size:1em}pre{display:block;margin-top:0;margin-bottom:1rem;overflow:auto;font-size:.875em}pre code{font-size:inherit;color:inherit;word-break:normal}code{font-size:.875em;color:var(--bs-code-color);word-wrap:break-word}a>code{color:inherit}kbd{padding:.1875rem .375rem;font-size:.875em;color:var(--bs-body-bg);background-color:var(--bs-body-color);border-radius:.25rem}kbd kbd{padding:0;font-size:1em}figure{margin:0 0 1rem}img,svg{vertical-align:middle}table{caption-side:bottom;border-collapse:collapse}caption{padding-top:.5rem;padding-bottom:.5rem;color:var(--bs-secondary-color);text-align:left}th{text-align:inherit;text-align:-webkit-match-parent}tbody,td,tfoot,th,thead,tr{border-color:inherit;border-style:solid;border-width:0}label{display:inline-block}button{border-radius:0}button:focus:not(:focus-visible){outline:0}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,select{text-transform:none}[role=button]{cursor:pointer}select{word-wrap:normal}select:disabled{opacity:1}[list]:not([type=date]):not([type=datetime-local]):not([type=month]):not([type=week]):not([type=time])::-webkit-calendar-picker-indicator{display:none!important}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]:not(:disabled),[type=reset]:not(:disabled),[type=submit]:not(:disabled),button:not(:disabled){cursor:pointer}::-moz-focus-inner{padding:0;border-style:none}textarea{resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{float:left;width:100%;padding:0;margin-bottom:.5rem;line-height:inherit;font-size:calc(1.275rem + .3vw)}@media (min-width:1200px){legend{font-size:1.5rem}}legend+*{clear:left}::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-fields-wrapper,::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-minute,::-webkit-datetime-edit-month-field,::-webkit-datetime-edit-text,::-webkit-datetime-edit-year-field{padding:0}::-webkit-inner-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-color-swatch-wrapper{padding:0}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}::file-selector-button{font:inherit;-webkit-appearance:button}output{display:inline-block}iframe{border:0}summary{display:list-item;cursor:pointer}progress{vertical-align:baseline}[hidden]{display:none!important}.lead{font-size:1.25rem;font-weight:300}.display-1{font-weight:300;line-height:1.2;font-size:calc(1.625rem + 4.5vw)}@media (min-width:1200px){.display-1{font-size:5rem}}.display-2{font-weight:300;line-height:1.2;font-size:calc(1.575rem + 3.9vw)}@media (min-width:1200px){.display-2{font-size:4.5rem}}.display-3{font-weight:300;line-height:1.2;font-size:calc(1.525rem + 3.3vw)}@media (min-width:1200px){.display-3{font-size:4rem}}.display-4{font-weight:300;line-height:1.2;font-size:calc(1.475rem + 2.7vw)}@media (min-width:1200px){.display-4{font-size:3.5rem}}.display-5{font-weight:300;line-height:1.2;font-size:calc(1.425rem + 2.1vw)}@media (min-width:1200px){.display-5{font-size:3rem}}.display-6{font-weight:300;line-height:1.2;font-size:calc(1.375rem + 1.5vw)}@media (min-width:1200px){.display-6{font-size:2.5rem}}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none}.list-inline-item{display:inline-block}.list-inline-item:not(:last-child){margin-right:.5rem}.initialism{font-size:.875em;text-transform:uppercase}.blockquote{margin-bottom:1rem;font-size:1.25rem}.blockquote>:last-child{margin-bottom:0}.blockquote-footer{margin-top:-1rem;margin-bottom:1rem;font-size:.875em;color:#6c757d}.blockquote-footer::before{content:"— "}.img-fluid{max-width:100%;height:auto}.img-thumbnail{padding:.25rem;background-color:var(--bs-body-bg);border:var(--bs-border-width) solid var(--bs-border-color);border-radius:var(--bs-border-radius);max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:.5rem;line-height:1}.figure-caption{font-size:.875em;color:var(--bs-secondary-color)}.container,.container-fluid,.container-lg,.container-md,.container-sm,.container-xl,.container-xxl{--bs-gutter-x:1.5rem;--bs-gutter-y:0;width:100%;padding-right:calc(var(--bs-gutter-x) * .5);padding-left:calc(var(--bs-gutter-x) * .5);margin-right:auto;margin-left:auto}@media (min-width:576px){.container,.container-sm{max-width:540px}}@media (min-width:768px){.container,.container-md,.container-sm{max-width:720px}}@media (min-width:992px){.container,.container-lg,.container-md,.container-sm{max-width:960px}}@media (min-width:1200px){.container,.container-lg,.container-md,.container-sm,.container-xl{max-width:1140px}}@media (min-width:1400px){.container,.container-lg,.container-md,.container-sm,.container-xl,.container-xxl{max-width:1320px}}:root{--bs-breakpoint-xs:0;--bs-breakpoint-sm:576px;--bs-breakpoint-md:768px;--bs-breakpoint-lg:992px;--bs-breakpoint-xl:1200px;--bs-breakpoint-xxl:1400px}.row{--bs-gutter-x:1.5rem;--bs-gutter-y:0;display:flex;flex-wrap:wrap;margin-top:calc(-1 * var(--bs-gutter-y));margin-right:calc(-.5 * var(--bs-gutter-x));margin-left:calc(-.5 * var(--bs-gutter-x))}.row>*{flex-shrink:0;width:100%;max-width:100%;padding-right:calc(var(--bs-gutter-x) * .5);padding-left:calc(var(--bs-gutter-x) * .5);margin-top:var(--bs-gutter-y)}.col{flex:1 0 0}.row-cols-auto>*{flex:0 0 auto;width:auto}.row-cols-1>*{flex:0 0 auto;width:100%}.row-cols-2>*{flex:0 0 auto;width:50%}.row-cols-3>*{flex:0 0 auto;width:33.33333333%}.row-cols-4>*{flex:0 0 auto;width:25%}.row-cols-5>*{flex:0 0 auto;width:20%}.row-cols-6>*{flex:0 0 auto;width:16.66666667%}.col-auto{flex:0 0 auto;width:auto}.col-1{flex:0 0 auto;width:8.33333333%}.col-2{flex:0 0 auto;width:16.66666667%}.col-3{flex:0 0 auto;width:25%}.col-4{flex:0 0 auto;width:33.33333333%}.col-5{flex:0 0 auto;width:41.66666667%}.col-6{flex:0 0 auto;width:50%}.col-7{flex:0 0 auto;width:58.33333333%}.col-8{flex:0 0 auto;width:66.66666667%}.col-9{flex:0 0 auto;width:75%}.col-10{flex:0 0 auto;width:83.33333333%}.col-11{flex:0 0 auto;width:91.66666667%}.col-12{flex:0 0 auto;width:100%}.offset-1{margin-left:8.33333333%}.offset-2{margin-left:16.66666667%}.offset-3{margin-left:25%}.offset-4{margin-left:33.33333333%}.offset-5{margin-left:41.66666667%}.offset-6{margin-left:50%}.offset-7{margin-left:58.33333333%}.offset-8{margin-left:66.66666667%}.offset-9{margin-left:75%}.offset-10{margin-left:83.33333333%}.offset-11{margin-left:91.66666667%}.g-0,.gx-0{--bs-gutter-x:0}.g-0,.gy-0{--bs-gutter-y:0}.g-1,.gx-1{--bs-gutter-x:0.25rem}.g-1,.gy-1{--bs-gutter-y:0.25rem}.g-2,.gx-2{--bs-gutter-x:0.5rem}.g-2,.gy-2{--bs-gutter-y:0.5rem}.g-3,.gx-3{--bs-gutter-x:1rem}.g-3,.gy-3{--bs-gutter-y:1rem}.g-4,.gx-4{--bs-gutter-x:1.5rem}.g-4,.gy-4{--bs-gutter-y:1.5rem}.g-5,.gx-5{--bs-gutter-x:3rem}.g-5,.gy-5{--bs-gutter-y:3rem}@media (min-width:576px){.col-sm{flex:1 0 0}.row-cols-sm-auto>*{flex:0 0 auto;width:auto}.row-cols-sm-1>*{flex:0 0 auto;width:100%}.row-cols-sm-2>*{flex:0 0 auto;width:50%}.row-cols-sm-3>*{flex:0 0 auto;width:33.33333333%}.row-cols-sm-4>*{flex:0 0 auto;width:25%}.row-cols-sm-5>*{flex:0 0 auto;width:20%}.row-cols-sm-6>*{flex:0 0 auto;width:16.66666667%}.col-sm-auto{flex:0 0 auto;width:auto}.col-sm-1{flex:0 0 auto;width:8.33333333%}.col-sm-2{flex:0 0 auto;width:16.66666667%}.col-sm-3{flex:0 0 auto;width:25%}.col-sm-4{flex:0 0 auto;width:33.33333333%}.col-sm-5{flex:0 0 auto;width:41.66666667%}.col-sm-6{flex:0 0 auto;width:50%}.col-sm-7{flex:0 0 auto;width:58.33333333%}.col-sm-8{flex:0 0 auto;width:66.66666667%}.col-sm-9{flex:0 0 auto;width:75%}.col-sm-10{flex:0 0 auto;width:83.33333333%}.col-sm-11{flex:0 0 auto;width:91.66666667%}.col-sm-12{flex:0 0 auto;width:100%}.offset-sm-0{margin-left:0}.offset-sm-1{margin-left:8.33333333%}.offset-sm-2{margin-left:16.66666667%}.offset-sm-3{margin-left:25%}.offset-sm-4{margin-left:33.33333333%}.offset-sm-5{margin-left:41.66666667%}.offset-sm-6{margin-left:50%}.offset-sm-7{margin-left:58.33333333%}.offset-sm-8{margin-left:66.66666667%}.offset-sm-9{margin-left:75%}.offset-sm-10{margin-left:83.33333333%}.offset-sm-11{margin-left:91.66666667%}.g-sm-0,.gx-sm-0{--bs-gutter-x:0}.g-sm-0,.gy-sm-0{--bs-gutter-y:0}.g-sm-1,.gx-sm-1{--bs-gutter-x:0.25rem}.g-sm-1,.gy-sm-1{--bs-gutter-y:0.25rem}.g-sm-2,.gx-sm-2{--bs-gutter-x:0.5rem}.g-sm-2,.gy-sm-2{--bs-gutter-y:0.5rem}.g-sm-3,.gx-sm-3{--bs-gutter-x:1rem}.g-sm-3,.gy-sm-3{--bs-gutter-y:1rem}.g-sm-4,.gx-sm-4{--bs-gutter-x:1.5rem}.g-sm-4,.gy-sm-4{--bs-gutter-y:1.5rem}.g-sm-5,.gx-sm-5{--bs-gutter-x:3rem}.g-sm-5,.gy-sm-5{--bs-gutter-y:3rem}}@media (min-width:768px){.col-md{flex:1 0 0}.row-cols-md-auto>*{flex:0 0 auto;width:auto}.row-cols-md-1>*{flex:0 0 auto;width:100%}.row-cols-md-2>*{flex:0 0 auto;width:50%}.row-cols-md-3>*{flex:0 0 auto;width:33.33333333%}.row-cols-md-4>*{flex:0 0 auto;width:25%}.row-cols-md-5>*{flex:0 0 auto;width:20%}.row-cols-md-6>*{flex:0 0 auto;width:16.66666667%}.col-md-auto{flex:0 0 auto;width:auto}.col-md-1{flex:0 0 auto;width:8.33333333%}.col-md-2{flex:0 0 auto;width:16.66666667%}.col-md-3{flex:0 0 auto;width:25%}.col-md-4{flex:0 0 auto;width:33.33333333%}.col-md-5{flex:0 0 auto;width:41.66666667%}.col-md-6{flex:0 0 auto;width:50%}.col-md-7{flex:0 0 auto;width:58.33333333%}.col-md-8{flex:0 0 auto;width:66.66666667%}.col-md-9{flex:0 0 auto;width:75%}.col-md-10{flex:0 0 auto;width:83.33333333%}.col-md-11{flex:0 0 auto;width:91.66666667%}.col-md-12{flex:0 0 auto;width:100%}.offset-md-0{margin-left:0}.offset-md-1{margin-left:8.33333333%}.offset-md-2{margin-left:16.66666667%}.offset-md-3{margin-left:25%}.offset-md-4{margin-left:33.33333333%}.offset-md-5{margin-left:41.66666667%}.offset-md-6{margin-left:50%}.offset-md-7{margin-left:58.33333333%}.offset-md-8{margin-left:66.66666667%}.offset-md-9{margin-left:75%}.offset-md-10{margin-left:83.33333333%}.offset-md-11{margin-left:91.66666667%}.g-md-0,.gx-md-0{--bs-gutter-x:0}.g-md-0,.gy-md-0{--bs-gutter-y:0}.g-md-1,.gx-md-1{--bs-gutter-x:0.25rem}.g-md-1,.gy-md-1{--bs-gutter-y:0.25rem}.g-md-2,.gx-md-2{--bs-gutter-x:0.5rem}.g-md-2,.gy-md-2{--bs-gutter-y:0.5rem}.g-md-3,.gx-md-3{--bs-gutter-x:1rem}.g-md-3,.gy-md-3{--bs-gutter-y:1rem}.g-md-4,.gx-md-4{--bs-gutter-x:1.5rem}.g-md-4,.gy-md-4{--bs-gutter-y:1.5rem}.g-md-5,.gx-md-5{--bs-gutter-x:3rem}.g-md-5,.gy-md-5{--bs-gutter-y:3rem}}@media (min-width:992px){.col-lg{flex:1 0 0}.row-cols-lg-auto>*{flex:0 0 auto;width:auto}.row-cols-lg-1>*{flex:0 0 auto;width:100%}.row-cols-lg-2>*{flex:0 0 auto;width:50%}.row-cols-lg-3>*{flex:0 0 auto;width:33.33333333%}.row-cols-lg-4>*{flex:0 0 auto;width:25%}.row-cols-lg-5>*{flex:0 0 auto;width:20%}.row-cols-lg-6>*{flex:0 0 auto;width:16.66666667%}.col-lg-auto{flex:0 0 auto;width:auto}.col-lg-1{flex:0 0 auto;width:8.33333333%}.col-lg-2{flex:0 0 auto;width:16.66666667%}.col-lg-3{flex:0 0 auto;width:25%}.col-lg-4{flex:0 0 auto;width:33.33333333%}.col-lg-5{flex:0 0 auto;width:41.66666667%}.col-lg-6{flex:0 0 auto;width:50%}.col-lg-7{flex:0 0 auto;width:58.33333333%}.col-lg-8{flex:0 0 auto;width:66.66666667%}.col-lg-9{flex:0 0 auto;width:75%}.col-lg-10{flex:0 0 auto;width:83.33333333%}.col-lg-11{flex:0 0 auto;width:91.66666667%}.col-lg-12{flex:0 0 auto;width:100%}.offset-lg-0{margin-left:0}.offset-lg-1{margin-left:8.33333333%}.offset-lg-2{margin-left:16.66666667%}.offset-lg-3{margin-left:25%}.offset-lg-4{margin-left:33.33333333%}.offset-lg-5{margin-left:41.66666667%}.offset-lg-6{margin-left:50%}.offset-lg-7{margin-left:58.33333333%}.offset-lg-8{margin-left:66.66666667%}.offset-lg-9{margin-left:75%}.offset-lg-10{margin-left:83.33333333%}.offset-lg-11{margin-left:91.66666667%}.g-lg-0,.gx-lg-0{--bs-gutter-x:0}.g-lg-0,.gy-lg-0{--bs-gutter-y:0}.g-lg-1,.gx-lg-1{--bs-gutter-x:0.25rem}.g-lg-1,.gy-lg-1{--bs-gutter-y:0.25rem}.g-lg-2,.gx-lg-2{--bs-gutter-x:0.5rem}.g-lg-2,.gy-lg-2{--bs-gutter-y:0.5rem}.g-lg-3,.gx-lg-3{--bs-gutter-x:1rem}.g-lg-3,.gy-lg-3{--bs-gutter-y:1rem}.g-lg-4,.gx-lg-4{--bs-gutter-x:1.5rem}.g-lg-4,.gy-lg-4{--bs-gutter-y:1.5rem}.g-lg-5,.gx-lg-5{--bs-gutter-x:3rem}.g-lg-5,.gy-lg-5{--bs-gutter-y:3rem}}@media (min-width:1200px){.col-xl{flex:1 0 0}.row-cols-xl-auto>*{flex:0 0 auto;width:auto}.row-cols-xl-1>*{flex:0 0 auto;width:100%}.row-cols-xl-2>*{flex:0 0 auto;width:50%}.row-cols-xl-3>*{flex:0 0 auto;width:33.33333333%}.row-cols-xl-4>*{flex:0 0 auto;width:25%}.row-cols-xl-5>*{flex:0 0 auto;width:20%}.row-cols-xl-6>*{flex:0 0 auto;width:16.66666667%}.col-xl-auto{flex:0 0 auto;width:auto}.col-xl-1{flex:0 0 auto;width:8.33333333%}.col-xl-2{flex:0 0 auto;width:16.66666667%}.col-xl-3{flex:0 0 auto;width:25%}.col-xl-4{flex:0 0 auto;width:33.33333333%}.col-xl-5{flex:0 0 auto;width:41.66666667%}.col-xl-6{flex:0 0 auto;width:50%}.col-xl-7{flex:0 0 auto;width:58.33333333%}.col-xl-8{flex:0 0 auto;width:66.66666667%}.col-xl-9{flex:0 0 auto;width:75%}.col-xl-10{flex:0 0 auto;width:83.33333333%}.col-xl-11{flex:0 0 auto;width:91.66666667%}.col-xl-12{flex:0 0 auto;width:100%}.offset-xl-0{margin-left:0}.offset-xl-1{margin-left:8.33333333%}.offset-xl-2{margin-left:16.66666667%}.offset-xl-3{margin-left:25%}.offset-xl-4{margin-left:33.33333333%}.offset-xl-5{margin-left:41.66666667%}.offset-xl-6{margin-left:50%}.offset-xl-7{margin-left:58.33333333%}.offset-xl-8{margin-left:66.66666667%}.offset-xl-9{margin-left:75%}.offset-xl-10{margin-left:83.33333333%}.offset-xl-11{margin-left:91.66666667%}.g-xl-0,.gx-xl-0{--bs-gutter-x:0}.g-xl-0,.gy-xl-0{--bs-gutter-y:0}.g-xl-1,.gx-xl-1{--bs-gutter-x:0.25rem}.g-xl-1,.gy-xl-1{--bs-gutter-y:0.25rem}.g-xl-2,.gx-xl-2{--bs-gutter-x:0.5rem}.g-xl-2,.gy-xl-2{--bs-gutter-y:0.5rem}.g-xl-3,.gx-xl-3{--bs-gutter-x:1rem}.g-xl-3,.gy-xl-3{--bs-gutter-y:1rem}.g-xl-4,.gx-xl-4{--bs-gutter-x:1.5rem}.g-xl-4,.gy-xl-4{--bs-gutter-y:1.5rem}.g-xl-5,.gx-xl-5{--bs-gutter-x:3rem}.g-xl-5,.gy-xl-5{--bs-gutter-y:3rem}}@media (min-width:1400px){.col-xxl{flex:1 0 0}.row-cols-xxl-auto>*{flex:0 0 auto;width:auto}.row-cols-xxl-1>*{flex:0 0 auto;width:100%}.row-cols-xxl-2>*{flex:0 0 auto;width:50%}.row-cols-xxl-3>*{flex:0 0 auto;width:33.33333333%}.row-cols-xxl-4>*{flex:0 0 auto;width:25%}.row-cols-xxl-5>*{flex:0 0 auto;width:20%}.row-cols-xxl-6>*{flex:0 0 auto;width:16.66666667%}.col-xxl-auto{flex:0 0 auto;width:auto}.col-xxl-1{flex:0 0 auto;width:8.33333333%}.col-xxl-2{flex:0 0 auto;width:16.66666667%}.col-xxl-3{flex:0 0 auto;width:25%}.col-xxl-4{flex:0 0 auto;width:33.33333333%}.col-xxl-5{flex:0 0 auto;width:41.66666667%}.col-xxl-6{flex:0 0 auto;width:50%}.col-xxl-7{flex:0 0 auto;width:58.33333333%}.col-xxl-8{flex:0 0 auto;width:66.66666667%}.col-xxl-9{flex:0 0 auto;width:75%}.col-xxl-10{flex:0 0 auto;width:83.33333333%}.col-xxl-11{flex:0 0 auto;width:91.66666667%}.col-xxl-12{flex:0 0 auto;width:100%}.offset-xxl-0{margin-left:0}.offset-xxl-1{margin-left:8.33333333%}.offset-xxl-2{margin-left:16.66666667%}.offset-xxl-3{margin-left:25%}.offset-xxl-4{margin-left:33.33333333%}.offset-xxl-5{margin-left:41.66666667%}.offset-xxl-6{margin-left:50%}.offset-xxl-7{margin-left:58.33333333%}.offset-xxl-8{margin-left:66.66666667%}.offset-xxl-9{margin-left:75%}.offset-xxl-10{margin-left:83.33333333%}.offset-xxl-11{margin-left:91.66666667%}.g-xxl-0,.gx-xxl-0{--bs-gutter-x:0}.g-xxl-0,.gy-xxl-0{--bs-gutter-y:0}.g-xxl-1,.gx-xxl-1{--bs-gutter-x:0.25rem}.g-xxl-1,.gy-xxl-1{--bs-gutter-y:0.25rem}.g-xxl-2,.gx-xxl-2{--bs-gutter-x:0.5rem}.g-xxl-2,.gy-xxl-2{--bs-gutter-y:0.5rem}.g-xxl-3,.gx-xxl-3{--bs-gutter-x:1rem}.g-xxl-3,.gy-xxl-3{--bs-gutter-y:1rem}.g-xxl-4,.gx-xxl-4{--bs-gutter-x:1.5rem}.g-xxl-4,.gy-xxl-4{--bs-gutter-y:1.5rem}.g-xxl-5,.gx-xxl-5{--bs-gutter-x:3rem}.g-xxl-5,.gy-xxl-5{--bs-gutter-y:3rem}}.table{--bs-table-color-type:initial;--bs-table-bg-type:initial;--bs-table-color-state:initial;--bs-table-bg-state:initial;--bs-table-color:var(--bs-emphasis-color);--bs-table-bg:var(--bs-body-bg);--bs-table-border-color:var(--bs-border-color);--bs-table-accent-bg:transparent;--bs-table-striped-color:var(--bs-emphasis-color);--bs-table-striped-bg:rgba(var(--bs-emphasis-color-rgb), 0.05);--bs-table-active-color:var(--bs-emphasis-color);--bs-table-active-bg:rgba(var(--bs-emphasis-color-rgb), 0.1);--bs-table-hover-color:var(--bs-emphasis-color);--bs-table-hover-bg:rgba(var(--bs-emphasis-color-rgb), 0.075);width:100%;margin-bottom:1rem;vertical-align:top;border-color:var(--bs-table-border-color)}.table>:not(caption)>*>*{padding:.5rem .5rem;color:var(--bs-table-color-state,var(--bs-table-color-type,var(--bs-table-color)));background-color:var(--bs-table-bg);border-bottom-width:var(--bs-border-width);box-shadow:inset 0 0 0 9999px var(--bs-table-bg-state,var(--bs-table-bg-type,var(--bs-table-accent-bg)))}.table>tbody{vertical-align:inherit}.table>thead{vertical-align:bottom}.table-group-divider{border-top:calc(var(--bs-border-width) * 2) solid currentcolor}.caption-top{caption-side:top}.table-sm>:not(caption)>*>*{padding:.25rem .25rem}.table-bordered>:not(caption)>*{border-width:var(--bs-border-width) 0}.table-bordered>:not(caption)>*>*{border-width:0 var(--bs-border-width)}.table-borderless>:not(caption)>*>*{border-bottom-width:0}.table-borderless>:not(:first-child){border-top-width:0}.table-striped>tbody>tr:nth-of-type(odd)>*{--bs-table-color-type:var(--bs-table-striped-color);--bs-table-bg-type:var(--bs-table-striped-bg)}.table-striped-columns>:not(caption)>tr>:nth-child(2n){--bs-table-color-type:var(--bs-table-striped-color);--bs-table-bg-type:var(--bs-table-striped-bg)}.table-active{--bs-table-color-state:var(--bs-table-active-color);--bs-table-bg-state:var(--bs-table-active-bg)}.table-hover>tbody>tr:hover>*{--bs-table-color-state:var(--bs-table-hover-color);--bs-table-bg-state:var(--bs-table-hover-bg)}.table-primary{--bs-table-color:#000;--bs-table-bg:#cfe2ff;--bs-table-border-color:#a6b5cc;--bs-table-striped-bg:#c5d7f2;--bs-table-striped-color:#000;--bs-table-active-bg:#bacbe6;--bs-table-active-color:#000;--bs-table-hover-bg:#bfd1ec;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-secondary{--bs-table-color:#000;--bs-table-bg:#e2e3e5;--bs-table-border-color:#b5b6b7;--bs-table-striped-bg:#d7d8da;--bs-table-striped-color:#000;--bs-table-active-bg:#cbccce;--bs-table-active-color:#000;--bs-table-hover-bg:#d1d2d4;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-success{--bs-table-color:#000;--bs-table-bg:#d1e7dd;--bs-table-border-color:#a7b9b1;--bs-table-striped-bg:#c7dbd2;--bs-table-striped-color:#000;--bs-table-active-bg:#bcd0c7;--bs-table-active-color:#000;--bs-table-hover-bg:#c1d6cc;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-info{--bs-table-color:#000;--bs-table-bg:#cff4fc;--bs-table-border-color:#a6c3ca;--bs-table-striped-bg:#c5e8ef;--bs-table-striped-color:#000;--bs-table-active-bg:#badce3;--bs-table-active-color:#000;--bs-table-hover-bg:#bfe2e9;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-warning{--bs-table-color:#000;--bs-table-bg:#fff3cd;--bs-table-border-color:#ccc2a4;--bs-table-striped-bg:#f2e7c3;--bs-table-striped-color:#000;--bs-table-active-bg:#e6dbb9;--bs-table-active-color:#000;--bs-table-hover-bg:#ece1be;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-danger{--bs-table-color:#000;--bs-table-bg:#f8d7da;--bs-table-border-color:#c6acae;--bs-table-striped-bg:#eccccf;--bs-table-striped-color:#000;--bs-table-active-bg:#dfc2c4;--bs-table-active-color:#000;--bs-table-hover-bg:#e5c7ca;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-light{--bs-table-color:#000;--bs-table-bg:#f8f9fa;--bs-table-border-color:#c6c7c8;--bs-table-striped-bg:#ecedee;--bs-table-striped-color:#000;--bs-table-active-bg:#dfe0e1;--bs-table-active-color:#000;--bs-table-hover-bg:#e5e6e7;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-dark{--bs-table-color:#fff;--bs-table-bg:#212529;--bs-table-border-color:#4d5154;--bs-table-striped-bg:#2c3034;--bs-table-striped-color:#fff;--bs-table-active-bg:#373b3e;--bs-table-active-color:#fff;--bs-table-hover-bg:#323539;--bs-table-hover-color:#fff;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-responsive{overflow-x:auto;-webkit-overflow-scrolling:touch}@media (max-width:575.98px){.table-responsive-sm{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:767.98px){.table-responsive-md{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:991.98px){.table-responsive-lg{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:1199.98px){.table-responsive-xl{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:1399.98px){.table-responsive-xxl{overflow-x:auto;-webkit-overflow-scrolling:touch}}.form-label{margin-bottom:.5rem}.col-form-label{padding-top:calc(.375rem + var(--bs-border-width));padding-bottom:calc(.375rem + var(--bs-border-width));margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(.5rem + var(--bs-border-width));padding-bottom:calc(.5rem + var(--bs-border-width));font-size:1.25rem}.col-form-label-sm{padding-top:calc(.25rem + var(--bs-border-width));padding-bottom:calc(.25rem + var(--bs-border-width));font-size:.875rem}.form-text{margin-top:.25rem;font-size:.875em;color:var(--bs-secondary-color)}.form-control{display:block;width:100%;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:var(--bs-body-color);-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:var(--bs-body-bg);background-clip:padding-box;border:var(--bs-border-width) solid var(--bs-border-color);border-radius:var(--bs-border-radius);transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-control{transition:none}}.form-control[type=file]{overflow:hidden}.form-control[type=file]:not(:disabled):not([readonly]){cursor:pointer}.form-control:focus{color:var(--bs-body-color);background-color:var(--bs-body-bg);border-color:#86b7fe;outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.form-control::-webkit-date-and-time-value{min-width:85px;height:1.5em;margin:0}.form-control::-webkit-datetime-edit{display:block;padding:0}.form-control::placeholder{color:var(--bs-secondary-color);opacity:1}.form-control:disabled{background-color:var(--bs-secondary-bg);opacity:1}.form-control::-webkit-file-upload-button{padding:.375rem .75rem;margin:-.375rem -.75rem;-webkit-margin-end:.75rem;margin-inline-end:.75rem;color:var(--bs-body-color);background-color:var(--bs-tertiary-bg);pointer-events:none;border-color:inherit;border-style:solid;border-width:0;border-inline-end-width:var(--bs-border-width);border-radius:0;-webkit-transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}.form-control::file-selector-button{padding:.375rem .75rem;margin:-.375rem -.75rem;-webkit-margin-end:.75rem;margin-inline-end:.75rem;color:var(--bs-body-color);background-color:var(--bs-tertiary-bg);pointer-events:none;border-color:inherit;border-style:solid;border-width:0;border-inline-end-width:var(--bs-border-width);border-radius:0;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-control::-webkit-file-upload-button{-webkit-transition:none;transition:none}.form-control::file-selector-button{transition:none}}.form-control:hover:not(:disabled):not([readonly])::-webkit-file-upload-button{background-color:var(--bs-secondary-bg)}.form-control:hover:not(:disabled):not([readonly])::file-selector-button{background-color:var(--bs-secondary-bg)}.form-control-plaintext{display:block;width:100%;padding:.375rem 0;margin-bottom:0;line-height:1.5;color:var(--bs-body-color);background-color:transparent;border:solid transparent;border-width:var(--bs-border-width) 0}.form-control-plaintext:focus{outline:0}.form-control-plaintext.form-control-lg,.form-control-plaintext.form-control-sm{padding-right:0;padding-left:0}.form-control-sm{min-height:calc(1.5em + .5rem + calc(var(--bs-border-width) * 2));padding:.25rem .5rem;font-size:.875rem;border-radius:var(--bs-border-radius-sm)}.form-control-sm::-webkit-file-upload-button{padding:.25rem .5rem;margin:-.25rem -.5rem;-webkit-margin-end:.5rem;margin-inline-end:.5rem}.form-control-sm::file-selector-button{padding:.25rem .5rem;margin:-.25rem -.5rem;-webkit-margin-end:.5rem;margin-inline-end:.5rem}.form-control-lg{min-height:calc(1.5em + 1rem + calc(var(--bs-border-width) * 2));padding:.5rem 1rem;font-size:1.25rem;border-radius:var(--bs-border-radius-lg)}.form-control-lg::-webkit-file-upload-button{padding:.5rem 1rem;margin:-.5rem -1rem;-webkit-margin-end:1rem;margin-inline-end:1rem}.form-control-lg::file-selector-button{padding:.5rem 1rem;margin:-.5rem -1rem;-webkit-margin-end:1rem;margin-inline-end:1rem}textarea.form-control{min-height:calc(1.5em + .75rem + calc(var(--bs-border-width) * 2))}textarea.form-control-sm{min-height:calc(1.5em + .5rem + calc(var(--bs-border-width) * 2))}textarea.form-control-lg{min-height:calc(1.5em + 1rem + calc(var(--bs-border-width) * 2))}.form-control-color{width:3rem;height:calc(1.5em + .75rem + calc(var(--bs-border-width) * 2));padding:.375rem}.form-control-color:not(:disabled):not([readonly]){cursor:pointer}.form-control-color::-moz-color-swatch{border:0!important;border-radius:var(--bs-border-radius)}.form-control-color::-webkit-color-swatch{border:0!important;border-radius:var(--bs-border-radius)}.form-control-color.form-control-sm{height:calc(1.5em + .5rem + calc(var(--bs-border-width) * 2))}.form-control-color.form-control-lg{height:calc(1.5em + 1rem + calc(var(--bs-border-width) * 2))}.form-select{--bs-form-select-bg-img:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23343a40' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/%3e%3c/svg%3e");display:block;width:100%;padding:.375rem 2.25rem .375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:var(--bs-body-color);-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:var(--bs-body-bg);background-image:var(--bs-form-select-bg-img),var(--bs-form-select-bg-icon,none);background-repeat:no-repeat;background-position:right .75rem center;background-size:16px 12px;border:var(--bs-border-width) solid var(--bs-border-color);border-radius:var(--bs-border-radius);transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-select{transition:none}}.form-select:focus{border-color:#86b7fe;outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.form-select[multiple],.form-select[size]:not([size="1"]){padding-right:.75rem;background-image:none}.form-select:disabled{background-color:var(--bs-secondary-bg)}.form-select:-moz-focusring{color:transparent;text-shadow:0 0 0 var(--bs-body-color)}.form-select-sm{padding-top:.25rem;padding-bottom:.25rem;padding-left:.5rem;font-size:.875rem;border-radius:var(--bs-border-radius-sm)}.form-select-lg{padding-top:.5rem;padding-bottom:.5rem;padding-left:1rem;font-size:1.25rem;border-radius:var(--bs-border-radius-lg)}[data-bs-theme=dark] .form-select{--bs-form-select-bg-img:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23dee2e6' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/%3e%3c/svg%3e")}.form-check{display:block;min-height:1.5rem;padding-left:1.5em;margin-bottom:.125rem}.form-check .form-check-input{float:left;margin-left:-1.5em}.form-check-reverse{padding-right:1.5em;padding-left:0;text-align:right}.form-check-reverse .form-check-input{float:right;margin-right:-1.5em;margin-left:0}.form-check-input{--bs-form-check-bg:var(--bs-body-bg);flex-shrink:0;width:1em;height:1em;margin-top:.25em;vertical-align:top;-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:var(--bs-form-check-bg);background-image:var(--bs-form-check-bg-image);background-repeat:no-repeat;background-position:center;background-size:contain;border:var(--bs-border-width) solid var(--bs-border-color);-webkit-print-color-adjust:exact;color-adjust:exact;print-color-adjust:exact}.form-check-input[type=checkbox]{border-radius:.25em}.form-check-input[type=radio]{border-radius:50%}.form-check-input:active{filter:brightness(90%)}.form-check-input:focus{border-color:#86b7fe;outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.form-check-input:checked{background-color:#0d6efd;border-color:#0d6efd}.form-check-input:checked[type=checkbox]{--bs-form-check-bg-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'%3e%3cpath fill='none' stroke='%23fff' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='m6 10 3 3 6-6'/%3e%3c/svg%3e")}.form-check-input:checked[type=radio]{--bs-form-check-bg-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='2' fill='%23fff'/%3e%3c/svg%3e")}.form-check-input[type=checkbox]:indeterminate{background-color:#0d6efd;border-color:#0d6efd;--bs-form-check-bg-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'%3e%3cpath fill='none' stroke='%23fff' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='M6 10h8'/%3e%3c/svg%3e")}.form-check-input:disabled{pointer-events:none;filter:none;opacity:.5}.form-check-input:disabled~.form-check-label,.form-check-input[disabled]~.form-check-label{cursor:default;opacity:.5}.form-switch{padding-left:2.5em}.form-switch .form-check-input{--bs-form-switch-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='rgba%280, 0, 0, 0.25%29'/%3e%3c/svg%3e");width:2em;margin-left:-2.5em;background-image:var(--bs-form-switch-bg);background-position:left center;border-radius:2em;transition:background-position .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-switch .form-check-input{transition:none}}.form-switch .form-check-input:focus{--bs-form-switch-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%2386b7fe'/%3e%3c/svg%3e")}.form-switch .form-check-input:checked{background-position:right center;--bs-form-switch-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23fff'/%3e%3c/svg%3e")}.form-switch.form-check-reverse{padding-right:2.5em;padding-left:0}.form-switch.form-check-reverse .form-check-input{margin-right:-2.5em;margin-left:0}.form-check-inline{display:inline-block;margin-right:1rem}.btn-check{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.btn-check:disabled+.btn,.btn-check[disabled]+.btn{pointer-events:none;filter:none;opacity:.65}[data-bs-theme=dark] .form-switch .form-check-input:not(:checked):not(:focus){--bs-form-switch-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='rgba%28255, 255, 255, 0.25%29'/%3e%3c/svg%3e")}.form-range{width:100%;height:1.5rem;padding:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:transparent}.form-range:focus{outline:0}.form-range:focus::-webkit-slider-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .25rem rgba(13,110,253,.25)}.form-range:focus::-moz-range-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .25rem rgba(13,110,253,.25)}.form-range::-moz-focus-outer{border:0}.form-range::-webkit-slider-thumb{width:1rem;height:1rem;margin-top:-.25rem;-webkit-appearance:none;appearance:none;background-color:#0d6efd;border:0;border-radius:1rem;-webkit-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-range::-webkit-slider-thumb{-webkit-transition:none;transition:none}}.form-range::-webkit-slider-thumb:active{background-color:#b6d4fe}.form-range::-webkit-slider-runnable-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:var(--bs-secondary-bg);border-color:transparent;border-radius:1rem}.form-range::-moz-range-thumb{width:1rem;height:1rem;-moz-appearance:none;appearance:none;background-color:#0d6efd;border:0;border-radius:1rem;-moz-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-range::-moz-range-thumb{-moz-transition:none;transition:none}}.form-range::-moz-range-thumb:active{background-color:#b6d4fe}.form-range::-moz-range-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:var(--bs-secondary-bg);border-color:transparent;border-radius:1rem}.form-range:disabled{pointer-events:none}.form-range:disabled::-webkit-slider-thumb{background-color:var(--bs-secondary-color)}.form-range:disabled::-moz-range-thumb{background-color:var(--bs-secondary-color)}.form-floating{position:relative}.form-floating>.form-control,.form-floating>.form-control-plaintext,.form-floating>.form-select{height:calc(3.5rem + calc(var(--bs-border-width) * 2));min-height:calc(3.5rem + calc(var(--bs-border-width) * 2));line-height:1.25}.form-floating>label{position:absolute;top:0;left:0;z-index:2;max-width:100%;height:100%;padding:1rem .75rem;overflow:hidden;color:rgba(var(--bs-body-color-rgb),.65);text-align:start;text-overflow:ellipsis;white-space:nowrap;pointer-events:none;border:var(--bs-border-width) solid transparent;transform-origin:0 0;transition:opacity .1s ease-in-out,transform .1s ease-in-out}@media (prefers-reduced-motion:reduce){.form-floating>label{transition:none}}.form-floating>.form-control,.form-floating>.form-control-plaintext{padding:1rem .75rem}.form-floating>.form-control-plaintext::placeholder,.form-floating>.form-control::placeholder{color:transparent}.form-floating>.form-control-plaintext:focus,.form-floating>.form-control-plaintext:not(:placeholder-shown),.form-floating>.form-control:focus,.form-floating>.form-control:not(:placeholder-shown){padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control-plaintext:-webkit-autofill,.form-floating>.form-control:-webkit-autofill{padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-select{padding-top:1.625rem;padding-bottom:.625rem;padding-left:.75rem}.form-floating>.form-control-plaintext~label,.form-floating>.form-control:focus~label,.form-floating>.form-control:not(:placeholder-shown)~label,.form-floating>.form-select~label{transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>.form-control:-webkit-autofill~label{transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>textarea:focus~label::after,.form-floating>textarea:not(:placeholder-shown)~label::after{position:absolute;inset:1rem 0.375rem;z-index:-1;height:1.5em;content:"";background-color:var(--bs-body-bg);border-radius:var(--bs-border-radius)}.form-floating>textarea:disabled~label::after{background-color:var(--bs-secondary-bg)}.form-floating>.form-control-plaintext~label{border-width:var(--bs-border-width) 0}.form-floating>.form-control:disabled~label,.form-floating>:disabled~label{color:#6c757d}.input-group{position:relative;display:flex;flex-wrap:wrap;align-items:stretch;width:100%}.input-group>.form-control,.input-group>.form-floating,.input-group>.form-select{position:relative;flex:1 1 auto;width:1%;min-width:0}.input-group>.form-control:focus,.input-group>.form-floating:focus-within,.input-group>.form-select:focus{z-index:5}.input-group .btn{position:relative;z-index:2}.input-group .btn:focus{z-index:5}.input-group-text{display:flex;align-items:center;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:var(--bs-body-color);text-align:center;white-space:nowrap;background-color:var(--bs-tertiary-bg);border:var(--bs-border-width) solid var(--bs-border-color);border-radius:var(--bs-border-radius)}.input-group-lg>.btn,.input-group-lg>.form-control,.input-group-lg>.form-select,.input-group-lg>.input-group-text{padding:.5rem 1rem;font-size:1.25rem;border-radius:var(--bs-border-radius-lg)}.input-group-sm>.btn,.input-group-sm>.form-control,.input-group-sm>.form-select,.input-group-sm>.input-group-text{padding:.25rem .5rem;font-size:.875rem;border-radius:var(--bs-border-radius-sm)}.input-group-lg>.form-select,.input-group-sm>.form-select{padding-right:3rem}.input-group:not(.has-validation)>.dropdown-toggle:nth-last-child(n+3),.input-group:not(.has-validation)>.form-floating:not(:last-child)>.form-control,.input-group:not(.has-validation)>.form-floating:not(:last-child)>.form-select,.input-group:not(.has-validation)>:not(:last-child):not(.dropdown-toggle):not(.dropdown-menu):not(.form-floating){border-top-right-radius:0;border-bottom-right-radius:0}.input-group.has-validation>.dropdown-toggle:nth-last-child(n+4),.input-group.has-validation>.form-floating:nth-last-child(n+3)>.form-control,.input-group.has-validation>.form-floating:nth-last-child(n+3)>.form-select,.input-group.has-validation>:nth-last-child(n+3):not(.dropdown-toggle):not(.dropdown-menu):not(.form-floating){border-top-right-radius:0;border-bottom-right-radius:0}.input-group>:not(:first-child):not(.dropdown-menu):not(.valid-tooltip):not(.valid-feedback):not(.invalid-tooltip):not(.invalid-feedback){margin-left:calc(-1 * var(--bs-border-width));border-top-left-radius:0;border-bottom-left-radius:0}.input-group>.form-floating:not(:first-child)>.form-control,.input-group>.form-floating:not(:first-child)>.form-select{border-top-left-radius:0;border-bottom-left-radius:0}.valid-feedback{display:none;width:100%;margin-top:.25rem;font-size:.875em;color:var(--bs-form-valid-color)}.valid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;color:#fff;background-color:var(--bs-success);border-radius:var(--bs-border-radius)}.is-valid~.valid-feedback,.is-valid~.valid-tooltip,.was-validated :valid~.valid-feedback,.was-validated :valid~.valid-tooltip{display:block}.form-control.is-valid,.was-validated .form-control:valid{border-color:var(--bs-form-valid-border-color);padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%23198754' d='M2.3 6.73.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-valid:focus,.was-validated .form-control:valid:focus{border-color:var(--bs-form-valid-border-color);box-shadow:0 0 0 .25rem rgba(var(--bs-success-rgb),.25)}.was-validated textarea.form-control:valid,textarea.form-control.is-valid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.form-select.is-valid,.was-validated .form-select:valid{border-color:var(--bs-form-valid-border-color)}.form-select.is-valid:not([multiple]):not([size]),.form-select.is-valid:not([multiple])[size="1"],.was-validated .form-select:valid:not([multiple]):not([size]),.was-validated .form-select:valid:not([multiple])[size="1"]{--bs-form-select-bg-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%23198754' d='M2.3 6.73.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1'/%3e%3c/svg%3e");padding-right:4.125rem;background-position:right .75rem center,center right 2.25rem;background-size:16px 12px,calc(.75em + .375rem) calc(.75em + .375rem)}.form-select.is-valid:focus,.was-validated .form-select:valid:focus{border-color:var(--bs-form-valid-border-color);box-shadow:0 0 0 .25rem rgba(var(--bs-success-rgb),.25)}.form-control-color.is-valid,.was-validated .form-control-color:valid{width:calc(3rem + calc(1.5em + .75rem))}.form-check-input.is-valid,.was-validated .form-check-input:valid{border-color:var(--bs-form-valid-border-color)}.form-check-input.is-valid:checked,.was-validated .form-check-input:valid:checked{background-color:var(--bs-form-valid-color)}.form-check-input.is-valid:focus,.was-validated .form-check-input:valid:focus{box-shadow:0 0 0 .25rem rgba(var(--bs-success-rgb),.25)}.form-check-input.is-valid~.form-check-label,.was-validated .form-check-input:valid~.form-check-label{color:var(--bs-form-valid-color)}.form-check-inline .form-check-input~.valid-feedback{margin-left:.5em}.input-group>.form-control:not(:focus).is-valid,.input-group>.form-floating:not(:focus-within).is-valid,.input-group>.form-select:not(:focus).is-valid,.was-validated .input-group>.form-control:not(:focus):valid,.was-validated .input-group>.form-floating:not(:focus-within):valid,.was-validated .input-group>.form-select:not(:focus):valid{z-index:3}.invalid-feedback{display:none;width:100%;margin-top:.25rem;font-size:.875em;color:var(--bs-form-invalid-color)}.invalid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;color:#fff;background-color:var(--bs-danger);border-radius:var(--bs-border-radius)}.is-invalid~.invalid-feedback,.is-invalid~.invalid-tooltip,.was-validated :invalid~.invalid-feedback,.was-validated :invalid~.invalid-tooltip{display:block}.form-control.is-invalid,.was-validated .form-control:invalid{border-color:var(--bs-form-invalid-border-color);padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 12 12' width='12' height='12' fill='none' stroke='%23dc3545'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23dc3545' stroke='none'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-invalid:focus,.was-validated .form-control:invalid:focus{border-color:var(--bs-form-invalid-border-color);box-shadow:0 0 0 .25rem rgba(var(--bs-danger-rgb),.25)}.was-validated textarea.form-control:invalid,textarea.form-control.is-invalid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.form-select.is-invalid,.was-validated .form-select:invalid{border-color:var(--bs-form-invalid-border-color)}.form-select.is-invalid:not([multiple]):not([size]),.form-select.is-invalid:not([multiple])[size="1"],.was-validated .form-select:invalid:not([multiple]):not([size]),.was-validated .form-select:invalid:not([multiple])[size="1"]{--bs-form-select-bg-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 12 12' width='12' height='12' fill='none' stroke='%23dc3545'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23dc3545' stroke='none'/%3e%3c/svg%3e");padding-right:4.125rem;background-position:right .75rem center,center right 2.25rem;background-size:16px 12px,calc(.75em + .375rem) calc(.75em + .375rem)}.form-select.is-invalid:focus,.was-validated .form-select:invalid:focus{border-color:var(--bs-form-invalid-border-color);box-shadow:0 0 0 .25rem rgba(var(--bs-danger-rgb),.25)}.form-control-color.is-invalid,.was-validated .form-control-color:invalid{width:calc(3rem + calc(1.5em + .75rem))}.form-check-input.is-invalid,.was-validated .form-check-input:invalid{border-color:var(--bs-form-invalid-border-color)}.form-check-input.is-invalid:checked,.was-validated .form-check-input:invalid:checked{background-color:var(--bs-form-invalid-color)}.form-check-input.is-invalid:focus,.was-validated .form-check-input:invalid:focus{box-shadow:0 0 0 .25rem rgba(var(--bs-danger-rgb),.25)}.form-check-input.is-invalid~.form-check-label,.was-validated .form-check-input:invalid~.form-check-label{color:var(--bs-form-invalid-color)}.form-check-inline .form-check-input~.invalid-feedback{margin-left:.5em}.input-group>.form-control:not(:focus).is-invalid,.input-group>.form-floating:not(:focus-within).is-invalid,.input-group>.form-select:not(:focus).is-invalid,.was-validated .input-group>.form-control:not(:focus):invalid,.was-validated .input-group>.form-floating:not(:focus-within):invalid,.was-validated .input-group>.form-select:not(:focus):invalid{z-index:4}.btn{--bs-btn-padding-x:0.75rem;--bs-btn-padding-y:0.375rem;--bs-btn-font-family: ;--bs-btn-font-size:1rem;--bs-btn-font-weight:400;--bs-btn-line-height:1.5;--bs-btn-color:var(--bs-body-color);--bs-btn-bg:transparent;--bs-btn-border-width:var(--bs-border-width);--bs-btn-border-color:transparent;--bs-btn-border-radius:var(--bs-border-radius);--bs-btn-hover-border-color:transparent;--bs-btn-box-shadow:inset 0 1px 0 rgba(255, 255, 255, 0.15),0 1px 1px rgba(0, 0, 0, 0.075);--bs-btn-disabled-opacity:0.65;--bs-btn-focus-box-shadow:0 0 0 0.25rem rgba(var(--bs-btn-focus-shadow-rgb), .5);display:inline-block;padding:var(--bs-btn-padding-y) var(--bs-btn-padding-x);font-family:var(--bs-btn-font-family);font-size:var(--bs-btn-font-size);font-weight:var(--bs-btn-font-weight);line-height:var(--bs-btn-line-height);color:var(--bs-btn-color);text-align:center;text-decoration:none;vertical-align:middle;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;user-select:none;border:var(--bs-btn-border-width) solid var(--bs-btn-border-color);border-radius:var(--bs-btn-border-radius);background-color:var(--bs-btn-bg);transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.btn{transition:none}}.btn:hover{color:var(--bs-btn-hover-color);background-color:var(--bs-btn-hover-bg);border-color:var(--bs-btn-hover-border-color)}.btn-check+.btn:hover{color:var(--bs-btn-color);background-color:var(--bs-btn-bg);border-color:var(--bs-btn-border-color)}.btn:focus-visible{color:var(--bs-btn-hover-color);background-color:var(--bs-btn-hover-bg);border-color:var(--bs-btn-hover-border-color);outline:0;box-shadow:var(--bs-btn-focus-box-shadow)}.btn-check:focus-visible+.btn{border-color:var(--bs-btn-hover-border-color);outline:0;box-shadow:var(--bs-btn-focus-box-shadow)}.btn-check:checked+.btn,.btn.active,.btn.show,.btn:first-child:active,:not(.btn-check)+.btn:active{color:var(--bs-btn-active-color);background-color:var(--bs-btn-active-bg);border-color:var(--bs-btn-active-border-color)}.btn-check:checked+.btn:focus-visible,.btn.active:focus-visible,.btn.show:focus-visible,.btn:first-child:active:focus-visible,:not(.btn-check)+.btn:active:focus-visible{box-shadow:var(--bs-btn-focus-box-shadow)}.btn-check:checked:focus-visible+.btn{box-shadow:var(--bs-btn-focus-box-shadow)}.btn.disabled,.btn:disabled,fieldset:disabled .btn{color:var(--bs-btn-disabled-color);pointer-events:none;background-color:var(--bs-btn-disabled-bg);border-color:var(--bs-btn-disabled-border-color);opacity:var(--bs-btn-disabled-opacity)}.btn-primary{--bs-btn-color:#fff;--bs-btn-bg:#0d6efd;--bs-btn-border-color:#0d6efd;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#0b5ed7;--bs-btn-hover-border-color:#0a58ca;--bs-btn-focus-shadow-rgb:49,132,253;--bs-btn-active-color:#fff;--bs-btn-active-bg:#0a58ca;--bs-btn-active-border-color:#0a53be;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#0d6efd;--bs-btn-disabled-border-color:#0d6efd}.btn-secondary{--bs-btn-color:#fff;--bs-btn-bg:#6c757d;--bs-btn-border-color:#6c757d;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#5c636a;--bs-btn-hover-border-color:#565e64;--bs-btn-focus-shadow-rgb:130,138,145;--bs-btn-active-color:#fff;--bs-btn-active-bg:#565e64;--bs-btn-active-border-color:#51585e;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#6c757d;--bs-btn-disabled-border-color:#6c757d}.btn-success{--bs-btn-color:#fff;--bs-btn-bg:#198754;--bs-btn-border-color:#198754;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#157347;--bs-btn-hover-border-color:#146c43;--bs-btn-focus-shadow-rgb:60,153,110;--bs-btn-active-color:#fff;--bs-btn-active-bg:#146c43;--bs-btn-active-border-color:#13653f;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#198754;--bs-btn-disabled-border-color:#198754}.btn-info{--bs-btn-color:#000;--bs-btn-bg:#0dcaf0;--bs-btn-border-color:#0dcaf0;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#31d2f2;--bs-btn-hover-border-color:#25cff2;--bs-btn-focus-shadow-rgb:11,172,204;--bs-btn-active-color:#000;--bs-btn-active-bg:#3dd5f3;--bs-btn-active-border-color:#25cff2;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#000;--bs-btn-disabled-bg:#0dcaf0;--bs-btn-disabled-border-color:#0dcaf0}.btn-warning{--bs-btn-color:#000;--bs-btn-bg:#ffc107;--bs-btn-border-color:#ffc107;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#ffca2c;--bs-btn-hover-border-color:#ffc720;--bs-btn-focus-shadow-rgb:217,164,6;--bs-btn-active-color:#000;--bs-btn-active-bg:#ffcd39;--bs-btn-active-border-color:#ffc720;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#000;--bs-btn-disabled-bg:#ffc107;--bs-btn-disabled-border-color:#ffc107}.btn-danger{--bs-btn-color:#fff;--bs-btn-bg:#dc3545;--bs-btn-border-color:#dc3545;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#bb2d3b;--bs-btn-hover-border-color:#b02a37;--bs-btn-focus-shadow-rgb:225,83,97;--bs-btn-active-color:#fff;--bs-btn-active-bg:#b02a37;--bs-btn-active-border-color:#a52834;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#dc3545;--bs-btn-disabled-border-color:#dc3545}.btn-light{--bs-btn-color:#000;--bs-btn-bg:#f8f9fa;--bs-btn-border-color:#f8f9fa;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#d3d4d5;--bs-btn-hover-border-color:#c6c7c8;--bs-btn-focus-shadow-rgb:211,212,213;--bs-btn-active-color:#000;--bs-btn-active-bg:#c6c7c8;--bs-btn-active-border-color:#babbbc;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#000;--bs-btn-disabled-bg:#f8f9fa;--bs-btn-disabled-border-color:#f8f9fa}.btn-dark{--bs-btn-color:#fff;--bs-btn-bg:#212529;--bs-btn-border-color:#212529;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#424649;--bs-btn-hover-border-color:#373b3e;--bs-btn-focus-shadow-rgb:66,70,73;--bs-btn-active-color:#fff;--bs-btn-active-bg:#4d5154;--bs-btn-active-border-color:#373b3e;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#212529;--bs-btn-disabled-border-color:#212529}.btn-outline-primary{--bs-btn-color:#0d6efd;--bs-btn-border-color:#0d6efd;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#0d6efd;--bs-btn-hover-border-color:#0d6efd;--bs-btn-focus-shadow-rgb:13,110,253;--bs-btn-active-color:#fff;--bs-btn-active-bg:#0d6efd;--bs-btn-active-border-color:#0d6efd;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#0d6efd;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#0d6efd;--bs-gradient:none}.btn-outline-secondary{--bs-btn-color:#6c757d;--bs-btn-border-color:#6c757d;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#6c757d;--bs-btn-hover-border-color:#6c757d;--bs-btn-focus-shadow-rgb:108,117,125;--bs-btn-active-color:#fff;--bs-btn-active-bg:#6c757d;--bs-btn-active-border-color:#6c757d;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#6c757d;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#6c757d;--bs-gradient:none}.btn-outline-success{--bs-btn-color:#198754;--bs-btn-border-color:#198754;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#198754;--bs-btn-hover-border-color:#198754;--bs-btn-focus-shadow-rgb:25,135,84;--bs-btn-active-color:#fff;--bs-btn-active-bg:#198754;--bs-btn-active-border-color:#198754;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#198754;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#198754;--bs-gradient:none}.btn-outline-info{--bs-btn-color:#0dcaf0;--bs-btn-border-color:#0dcaf0;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#0dcaf0;--bs-btn-hover-border-color:#0dcaf0;--bs-btn-focus-shadow-rgb:13,202,240;--bs-btn-active-color:#000;--bs-btn-active-bg:#0dcaf0;--bs-btn-active-border-color:#0dcaf0;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#0dcaf0;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#0dcaf0;--bs-gradient:none}.btn-outline-warning{--bs-btn-color:#ffc107;--bs-btn-border-color:#ffc107;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#ffc107;--bs-btn-hover-border-color:#ffc107;--bs-btn-focus-shadow-rgb:255,193,7;--bs-btn-active-color:#000;--bs-btn-active-bg:#ffc107;--bs-btn-active-border-color:#ffc107;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#ffc107;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#ffc107;--bs-gradient:none}.btn-outline-danger{--bs-btn-color:#dc3545;--bs-btn-border-color:#dc3545;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#dc3545;--bs-btn-hover-border-color:#dc3545;--bs-btn-focus-shadow-rgb:220,53,69;--bs-btn-active-color:#fff;--bs-btn-active-bg:#dc3545;--bs-btn-active-border-color:#dc3545;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#dc3545;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#dc3545;--bs-gradient:none}.btn-outline-light{--bs-btn-color:#f8f9fa;--bs-btn-border-color:#f8f9fa;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#f8f9fa;--bs-btn-hover-border-color:#f8f9fa;--bs-btn-focus-shadow-rgb:248,249,250;--bs-btn-active-color:#000;--bs-btn-active-bg:#f8f9fa;--bs-btn-active-border-color:#f8f9fa;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#f8f9fa;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#f8f9fa;--bs-gradient:none}.btn-outline-dark{--bs-btn-color:#212529;--bs-btn-border-color:#212529;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#212529;--bs-btn-hover-border-color:#212529;--bs-btn-focus-shadow-rgb:33,37,41;--bs-btn-active-color:#fff;--bs-btn-active-bg:#212529;--bs-btn-active-border-color:#212529;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#212529;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#212529;--bs-gradient:none}.btn-link{--bs-btn-font-weight:400;--bs-btn-color:var(--bs-link-color);--bs-btn-bg:transparent;--bs-btn-border-color:transparent;--bs-btn-hover-color:var(--bs-link-hover-color);--bs-btn-hover-border-color:transparent;--bs-btn-active-color:var(--bs-link-hover-color);--bs-btn-active-border-color:transparent;--bs-btn-disabled-color:#6c757d;--bs-btn-disabled-border-color:transparent;--bs-btn-box-shadow:0 0 0 #000;--bs-btn-focus-shadow-rgb:49,132,253;text-decoration:underline}.btn-link:focus-visible{color:var(--bs-btn-color)}.btn-link:hover{color:var(--bs-btn-hover-color)}.btn-group-lg>.btn,.btn-lg{--bs-btn-padding-y:0.5rem;--bs-btn-padding-x:1rem;--bs-btn-font-size:1.25rem;--bs-btn-border-radius:var(--bs-border-radius-lg)}.btn-group-sm>.btn,.btn-sm{--bs-btn-padding-y:0.25rem;--bs-btn-padding-x:0.5rem;--bs-btn-font-size:0.875rem;--bs-btn-border-radius:var(--bs-border-radius-sm)}.fade{transition:opacity .15s linear}@media (prefers-reduced-motion:reduce){.fade{transition:none}}.fade:not(.show){opacity:0}.collapse:not(.show){display:none}.collapsing{height:0;overflow:hidden;transition:height .35s ease}@media (prefers-reduced-motion:reduce){.collapsing{transition:none}}.collapsing.collapse-horizontal{width:0;height:auto;transition:width .35s ease}@media (prefers-reduced-motion:reduce){.collapsing.collapse-horizontal{transition:none}}.dropdown,.dropdown-center,.dropend,.dropstart,.dropup,.dropup-center{position:relative}.dropdown-toggle{white-space:nowrap}.dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid;border-right:.3em solid transparent;border-bottom:0;border-left:.3em solid transparent}.dropdown-toggle:empty::after{margin-left:0}.dropdown-menu{--bs-dropdown-zindex:1000;--bs-dropdown-min-width:10rem;--bs-dropdown-padding-x:0;--bs-dropdown-padding-y:0.5rem;--bs-dropdown-spacer:0.125rem;--bs-dropdown-font-size:1rem;--bs-dropdown-color:var(--bs-body-color);--bs-dropdown-bg:var(--bs-body-bg);--bs-dropdown-border-color:var(--bs-border-color-translucent);--bs-dropdown-border-radius:var(--bs-border-radius);--bs-dropdown-border-width:var(--bs-border-width);--bs-dropdown-inner-border-radius:calc(var(--bs-border-radius) - var(--bs-border-width));--bs-dropdown-divider-bg:var(--bs-border-color-translucent);--bs-dropdown-divider-margin-y:0.5rem;--bs-dropdown-box-shadow:var(--bs-box-shadow);--bs-dropdown-link-color:var(--bs-body-color);--bs-dropdown-link-hover-color:var(--bs-body-color);--bs-dropdown-link-hover-bg:var(--bs-tertiary-bg);--bs-dropdown-link-active-color:#fff;--bs-dropdown-link-active-bg:#0d6efd;--bs-dropdown-link-disabled-color:var(--bs-tertiary-color);--bs-dropdown-item-padding-x:1rem;--bs-dropdown-item-padding-y:0.25rem;--bs-dropdown-header-color:#6c757d;--bs-dropdown-header-padding-x:1rem;--bs-dropdown-header-padding-y:0.5rem;position:absolute;z-index:var(--bs-dropdown-zindex);display:none;min-width:var(--bs-dropdown-min-width);padding:var(--bs-dropdown-padding-y) var(--bs-dropdown-padding-x);margin:0;font-size:var(--bs-dropdown-font-size);color:var(--bs-dropdown-color);text-align:left;list-style:none;background-color:var(--bs-dropdown-bg);background-clip:padding-box;border:var(--bs-dropdown-border-width) solid var(--bs-dropdown-border-color);border-radius:var(--bs-dropdown-border-radius)}.dropdown-menu[data-bs-popper]{top:100%;left:0;margin-top:var(--bs-dropdown-spacer)}.dropdown-menu-start{--bs-position:start}.dropdown-menu-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-end{--bs-position:end}.dropdown-menu-end[data-bs-popper]{right:0;left:auto}@media (min-width:576px){.dropdown-menu-sm-start{--bs-position:start}.dropdown-menu-sm-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-sm-end{--bs-position:end}.dropdown-menu-sm-end[data-bs-popper]{right:0;left:auto}}@media (min-width:768px){.dropdown-menu-md-start{--bs-position:start}.dropdown-menu-md-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-md-end{--bs-position:end}.dropdown-menu-md-end[data-bs-popper]{right:0;left:auto}}@media (min-width:992px){.dropdown-menu-lg-start{--bs-position:start}.dropdown-menu-lg-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-lg-end{--bs-position:end}.dropdown-menu-lg-end[data-bs-popper]{right:0;left:auto}}@media (min-width:1200px){.dropdown-menu-xl-start{--bs-position:start}.dropdown-menu-xl-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-xl-end{--bs-position:end}.dropdown-menu-xl-end[data-bs-popper]{right:0;left:auto}}@media (min-width:1400px){.dropdown-menu-xxl-start{--bs-position:start}.dropdown-menu-xxl-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-xxl-end{--bs-position:end}.dropdown-menu-xxl-end[data-bs-popper]{right:0;left:auto}}.dropup .dropdown-menu[data-bs-popper]{top:auto;bottom:100%;margin-top:0;margin-bottom:var(--bs-dropdown-spacer)}.dropup .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:0;border-right:.3em solid transparent;border-bottom:.3em solid;border-left:.3em solid transparent}.dropup .dropdown-toggle:empty::after{margin-left:0}.dropend .dropdown-menu[data-bs-popper]{top:0;right:auto;left:100%;margin-top:0;margin-left:var(--bs-dropdown-spacer)}.dropend .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:0;border-bottom:.3em solid transparent;border-left:.3em solid}.dropend .dropdown-toggle:empty::after{margin-left:0}.dropend .dropdown-toggle::after{vertical-align:0}.dropstart .dropdown-menu[data-bs-popper]{top:0;right:100%;left:auto;margin-top:0;margin-right:var(--bs-dropdown-spacer)}.dropstart .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:""}.dropstart .dropdown-toggle::after{display:none}.dropstart .dropdown-toggle::before{display:inline-block;margin-right:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:.3em solid;border-bottom:.3em solid transparent}.dropstart .dropdown-toggle:empty::after{margin-left:0}.dropstart .dropdown-toggle::before{vertical-align:0}.dropdown-divider{height:0;margin:var(--bs-dropdown-divider-margin-y) 0;overflow:hidden;border-top:1px solid var(--bs-dropdown-divider-bg);opacity:1}.dropdown-item{display:block;width:100%;padding:var(--bs-dropdown-item-padding-y) var(--bs-dropdown-item-padding-x);clear:both;font-weight:400;color:var(--bs-dropdown-link-color);text-align:inherit;text-decoration:none;white-space:nowrap;background-color:transparent;border:0;border-radius:var(--bs-dropdown-item-border-radius,0)}.dropdown-item:focus,.dropdown-item:hover{color:var(--bs-dropdown-link-hover-color);background-color:var(--bs-dropdown-link-hover-bg)}.dropdown-item.active,.dropdown-item:active{color:var(--bs-dropdown-link-active-color);text-decoration:none;background-color:var(--bs-dropdown-link-active-bg)}.dropdown-item.disabled,.dropdown-item:disabled{color:var(--bs-dropdown-link-disabled-color);pointer-events:none;background-color:transparent}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:var(--bs-dropdown-header-padding-y) var(--bs-dropdown-header-padding-x);margin-bottom:0;font-size:.875rem;color:var(--bs-dropdown-header-color);white-space:nowrap}.dropdown-item-text{display:block;padding:var(--bs-dropdown-item-padding-y) var(--bs-dropdown-item-padding-x);color:var(--bs-dropdown-link-color)}.dropdown-menu-dark{--bs-dropdown-color:#dee2e6;--bs-dropdown-bg:#343a40;--bs-dropdown-border-color:var(--bs-border-color-translucent);--bs-dropdown-box-shadow: ;--bs-dropdown-link-color:#dee2e6;--bs-dropdown-link-hover-color:#fff;--bs-dropdown-divider-bg:var(--bs-border-color-translucent);--bs-dropdown-link-hover-bg:rgba(255, 255, 255, 0.15);--bs-dropdown-link-active-color:#fff;--bs-dropdown-link-active-bg:#0d6efd;--bs-dropdown-link-disabled-color:#adb5bd;--bs-dropdown-header-color:#adb5bd}.btn-group,.btn-group-vertical{position:relative;display:inline-flex;vertical-align:middle}.btn-group-vertical>.btn,.btn-group>.btn{position:relative;flex:1 1 auto}.btn-group-vertical>.btn-check:checked+.btn,.btn-group-vertical>.btn-check:focus+.btn,.btn-group-vertical>.btn.active,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn:focus,.btn-group-vertical>.btn:hover,.btn-group>.btn-check:checked+.btn,.btn-group>.btn-check:focus+.btn,.btn-group>.btn.active,.btn-group>.btn:active,.btn-group>.btn:focus,.btn-group>.btn:hover{z-index:1}.btn-toolbar{display:flex;flex-wrap:wrap;justify-content:flex-start}.btn-toolbar .input-group{width:auto}.btn-group{border-radius:var(--bs-border-radius)}.btn-group>.btn-group:not(:first-child),.btn-group>:not(.btn-check:first-child)+.btn{margin-left:calc(-1 * var(--bs-border-width))}.btn-group>.btn-group:not(:last-child)>.btn,.btn-group>.btn.dropdown-toggle-split:first-child,.btn-group>.btn:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:not(:first-child)>.btn,.btn-group>.btn:nth-child(n+3),.btn-group>:not(.btn-check)+.btn{border-top-left-radius:0;border-bottom-left-radius:0}.dropdown-toggle-split{padding-right:.5625rem;padding-left:.5625rem}.dropdown-toggle-split::after,.dropend .dropdown-toggle-split::after,.dropup .dropdown-toggle-split::after{margin-left:0}.dropstart .dropdown-toggle-split::before{margin-right:0}.btn-group-sm>.btn+.dropdown-toggle-split,.btn-sm+.dropdown-toggle-split{padding-right:.375rem;padding-left:.375rem}.btn-group-lg>.btn+.dropdown-toggle-split,.btn-lg+.dropdown-toggle-split{padding-right:.75rem;padding-left:.75rem}.btn-group-vertical{flex-direction:column;align-items:flex-start;justify-content:center}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group{width:100%}.btn-group-vertical>.btn-group:not(:first-child),.btn-group-vertical>.btn:not(:first-child){margin-top:calc(-1 * var(--bs-border-width))}.btn-group-vertical>.btn-group:not(:last-child)>.btn,.btn-group-vertical>.btn:not(:last-child):not(.dropdown-toggle){border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:not(:first-child)>.btn,.btn-group-vertical>.btn:nth-child(n+3),.btn-group-vertical>:not(.btn-check)+.btn{border-top-left-radius:0;border-top-right-radius:0}.nav{--bs-nav-link-padding-x:1rem;--bs-nav-link-padding-y:0.5rem;--bs-nav-link-font-weight: ;--bs-nav-link-color:var(--bs-link-color);--bs-nav-link-hover-color:var(--bs-link-hover-color);--bs-nav-link-disabled-color:var(--bs-secondary-color);display:flex;flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:var(--bs-nav-link-padding-y) var(--bs-nav-link-padding-x);font-size:var(--bs-nav-link-font-size);font-weight:var(--bs-nav-link-font-weight);color:var(--bs-nav-link-color);text-decoration:none;background:0 0;border:0;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out}@media (prefers-reduced-motion:reduce){.nav-link{transition:none}}.nav-link:focus,.nav-link:hover{color:var(--bs-nav-link-hover-color)}.nav-link:focus-visible{outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.nav-link.disabled,.nav-link:disabled{color:var(--bs-nav-link-disabled-color);pointer-events:none;cursor:default}.nav-tabs{--bs-nav-tabs-border-width:var(--bs-border-width);--bs-nav-tabs-border-color:var(--bs-border-color);--bs-nav-tabs-border-radius:var(--bs-border-radius);--bs-nav-tabs-link-hover-border-color:var(--bs-secondary-bg) var(--bs-secondary-bg) var(--bs-border-color);--bs-nav-tabs-link-active-color:var(--bs-emphasis-color);--bs-nav-tabs-link-active-bg:var(--bs-body-bg);--bs-nav-tabs-link-active-border-color:var(--bs-border-color) var(--bs-border-color) var(--bs-body-bg);border-bottom:var(--bs-nav-tabs-border-width) solid var(--bs-nav-tabs-border-color)}.nav-tabs .nav-link{margin-bottom:calc(-1 * var(--bs-nav-tabs-border-width));border:var(--bs-nav-tabs-border-width) solid transparent;border-top-left-radius:var(--bs-nav-tabs-border-radius);border-top-right-radius:var(--bs-nav-tabs-border-radius)}.nav-tabs .nav-link:focus,.nav-tabs .nav-link:hover{isolation:isolate;border-color:var(--bs-nav-tabs-link-hover-border-color)}.nav-tabs .nav-item.show .nav-link,.nav-tabs .nav-link.active{color:var(--bs-nav-tabs-link-active-color);background-color:var(--bs-nav-tabs-link-active-bg);border-color:var(--bs-nav-tabs-link-active-border-color)}.nav-tabs .dropdown-menu{margin-top:calc(-1 * var(--bs-nav-tabs-border-width));border-top-left-radius:0;border-top-right-radius:0}.nav-pills{--bs-nav-pills-border-radius:var(--bs-border-radius);--bs-nav-pills-link-active-color:#fff;--bs-nav-pills-link-active-bg:#0d6efd}.nav-pills .nav-link{border-radius:var(--bs-nav-pills-border-radius)}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:var(--bs-nav-pills-link-active-color);background-color:var(--bs-nav-pills-link-active-bg)}.nav-underline{--bs-nav-underline-gap:1rem;--bs-nav-underline-border-width:0.125rem;--bs-nav-underline-link-active-color:var(--bs-emphasis-color);gap:var(--bs-nav-underline-gap)}.nav-underline .nav-link{padding-right:0;padding-left:0;border-bottom:var(--bs-nav-underline-border-width) solid transparent}.nav-underline .nav-link:focus,.nav-underline .nav-link:hover{border-bottom-color:currentcolor}.nav-underline .nav-link.active,.nav-underline .show>.nav-link{font-weight:700;color:var(--bs-nav-underline-link-active-color);border-bottom-color:currentcolor}.nav-fill .nav-item,.nav-fill>.nav-link{flex:1 1 auto;text-align:center}.nav-justified .nav-item,.nav-justified>.nav-link{flex-grow:1;flex-basis:0;text-align:center}.nav-fill .nav-item .nav-link,.nav-justified .nav-item .nav-link{width:100%}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.navbar{--bs-navbar-padding-x:0;--bs-navbar-padding-y:0.5rem;--bs-navbar-color:rgba(var(--bs-emphasis-color-rgb), 0.65);--bs-navbar-hover-color:rgba(var(--bs-emphasis-color-rgb), 0.8);--bs-navbar-disabled-color:rgba(var(--bs-emphasis-color-rgb), 0.3);--bs-navbar-active-color:rgba(var(--bs-emphasis-color-rgb), 1);--bs-navbar-brand-padding-y:0.3125rem;--bs-navbar-brand-margin-end:1rem;--bs-navbar-brand-font-size:1.25rem;--bs-navbar-brand-color:rgba(var(--bs-emphasis-color-rgb), 1);--bs-navbar-brand-hover-color:rgba(var(--bs-emphasis-color-rgb), 1);--bs-navbar-nav-link-padding-x:0.5rem;--bs-navbar-toggler-padding-y:0.25rem;--bs-navbar-toggler-padding-x:0.75rem;--bs-navbar-toggler-font-size:1.25rem;--bs-navbar-toggler-icon-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%2833, 37, 41, 0.75%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e");--bs-navbar-toggler-border-color:rgba(var(--bs-emphasis-color-rgb), 0.15);--bs-navbar-toggler-border-radius:var(--bs-border-radius);--bs-navbar-toggler-focus-width:0.25rem;--bs-navbar-toggler-transition:box-shadow 0.15s ease-in-out;position:relative;display:flex;flex-wrap:wrap;align-items:center;justify-content:space-between;padding:var(--bs-navbar-padding-y) var(--bs-navbar-padding-x)}.navbar>.container,.navbar>.container-fluid,.navbar>.container-lg,.navbar>.container-md,.navbar>.container-sm,.navbar>.container-xl,.navbar>.container-xxl{display:flex;flex-wrap:inherit;align-items:center;justify-content:space-between}.navbar-brand{padding-top:var(--bs-navbar-brand-padding-y);padding-bottom:var(--bs-navbar-brand-padding-y);margin-right:var(--bs-navbar-brand-margin-end);font-size:var(--bs-navbar-brand-font-size);color:var(--bs-navbar-brand-color);text-decoration:none;white-space:nowrap}.navbar-brand:focus,.navbar-brand:hover{color:var(--bs-navbar-brand-hover-color)}.navbar-nav{--bs-nav-link-padding-x:0;--bs-nav-link-padding-y:0.5rem;--bs-nav-link-font-weight: ;--bs-nav-link-color:var(--bs-navbar-color);--bs-nav-link-hover-color:var(--bs-navbar-hover-color);--bs-nav-link-disabled-color:var(--bs-navbar-disabled-color);display:flex;flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link.active,.navbar-nav .nav-link.show{color:var(--bs-navbar-active-color)}.navbar-nav .dropdown-menu{position:static}.navbar-text{padding-top:.5rem;padding-bottom:.5rem;color:var(--bs-navbar-color)}.navbar-text a,.navbar-text a:focus,.navbar-text a:hover{color:var(--bs-navbar-active-color)}.navbar-collapse{flex-grow:1;flex-basis:100%;align-items:center}.navbar-toggler{padding:var(--bs-navbar-toggler-padding-y) var(--bs-navbar-toggler-padding-x);font-size:var(--bs-navbar-toggler-font-size);line-height:1;color:var(--bs-navbar-color);background-color:transparent;border:var(--bs-border-width) solid var(--bs-navbar-toggler-border-color);border-radius:var(--bs-navbar-toggler-border-radius);transition:var(--bs-navbar-toggler-transition)}@media (prefers-reduced-motion:reduce){.navbar-toggler{transition:none}}.navbar-toggler:hover{text-decoration:none}.navbar-toggler:focus{text-decoration:none;outline:0;box-shadow:0 0 0 var(--bs-navbar-toggler-focus-width)}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;background-image:var(--bs-navbar-toggler-icon-bg);background-repeat:no-repeat;background-position:center;background-size:100%}.navbar-nav-scroll{max-height:var(--bs-scroll-height,75vh);overflow-y:auto}@media (min-width:576px){.navbar-expand-sm{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-sm .navbar-nav{flex-direction:row}.navbar-expand-sm .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-sm .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-sm .navbar-nav-scroll{overflow:visible}.navbar-expand-sm .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-sm .navbar-toggler{display:none}.navbar-expand-sm .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-sm .offcanvas .offcanvas-header{display:none}.navbar-expand-sm .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:768px){.navbar-expand-md{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-md .navbar-nav{flex-direction:row}.navbar-expand-md .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-md .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-md .navbar-nav-scroll{overflow:visible}.navbar-expand-md .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-md .navbar-toggler{display:none}.navbar-expand-md .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-md .offcanvas .offcanvas-header{display:none}.navbar-expand-md .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:992px){.navbar-expand-lg{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-lg .navbar-nav{flex-direction:row}.navbar-expand-lg .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-lg .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-lg .navbar-nav-scroll{overflow:visible}.navbar-expand-lg .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-lg .navbar-toggler{display:none}.navbar-expand-lg .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-lg .offcanvas .offcanvas-header{display:none}.navbar-expand-lg .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:1200px){.navbar-expand-xl{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-xl .navbar-nav{flex-direction:row}.navbar-expand-xl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xl .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-xl .navbar-nav-scroll{overflow:visible}.navbar-expand-xl .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-xl .navbar-toggler{display:none}.navbar-expand-xl .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-xl .offcanvas .offcanvas-header{display:none}.navbar-expand-xl .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:1400px){.navbar-expand-xxl{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-xxl .navbar-nav{flex-direction:row}.navbar-expand-xxl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xxl .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-xxl .navbar-nav-scroll{overflow:visible}.navbar-expand-xxl .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-xxl .navbar-toggler{display:none}.navbar-expand-xxl .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-xxl .offcanvas .offcanvas-header{display:none}.navbar-expand-xxl .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}.navbar-expand{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand .navbar-nav{flex-direction:row}.navbar-expand .navbar-nav .dropdown-menu{position:absolute}.navbar-expand .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand .navbar-nav-scroll{overflow:visible}.navbar-expand .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-expand .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand .offcanvas .offcanvas-header{display:none}.navbar-expand .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}.navbar-dark,.navbar[data-bs-theme=dark]{--bs-navbar-color:rgba(255, 255, 255, 0.55);--bs-navbar-hover-color:rgba(255, 255, 255, 0.75);--bs-navbar-disabled-color:rgba(255, 255, 255, 0.25);--bs-navbar-active-color:#fff;--bs-navbar-brand-color:#fff;--bs-navbar-brand-hover-color:#fff;--bs-navbar-toggler-border-color:rgba(255, 255, 255, 0.1);--bs-navbar-toggler-icon-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}[data-bs-theme=dark] .navbar-toggler-icon{--bs-navbar-toggler-icon-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.card{--bs-card-spacer-y:1rem;--bs-card-spacer-x:1rem;--bs-card-title-spacer-y:0.5rem;--bs-card-title-color: ;--bs-card-subtitle-color: ;--bs-card-border-width:var(--bs-border-width);--bs-card-border-color:var(--bs-border-color-translucent);--bs-card-border-radius:var(--bs-border-radius);--bs-card-box-shadow: ;--bs-card-inner-border-radius:calc(var(--bs-border-radius) - (var(--bs-border-width)));--bs-card-cap-padding-y:0.5rem;--bs-card-cap-padding-x:1rem;--bs-card-cap-bg:rgba(var(--bs-body-color-rgb), 0.03);--bs-card-cap-color: ;--bs-card-height: ;--bs-card-color: ;--bs-card-bg:var(--bs-body-bg);--bs-card-img-overlay-padding:1rem;--bs-card-group-margin:0.75rem;position:relative;display:flex;flex-direction:column;min-width:0;height:var(--bs-card-height);color:var(--bs-body-color);word-wrap:break-word;background-color:var(--bs-card-bg);background-clip:border-box;border:var(--bs-card-border-width) solid var(--bs-card-border-color);border-radius:var(--bs-card-border-radius)}.card>hr{margin-right:0;margin-left:0}.card>.list-group{border-top:inherit;border-bottom:inherit}.card>.list-group:first-child{border-top-width:0;border-top-left-radius:var(--bs-card-inner-border-radius);border-top-right-radius:var(--bs-card-inner-border-radius)}.card>.list-group:last-child{border-bottom-width:0;border-bottom-right-radius:var(--bs-card-inner-border-radius);border-bottom-left-radius:var(--bs-card-inner-border-radius)}.card>.card-header+.list-group,.card>.list-group+.card-footer{border-top:0}.card-body{flex:1 1 auto;padding:var(--bs-card-spacer-y) var(--bs-card-spacer-x);color:var(--bs-card-color)}.card-title{margin-bottom:var(--bs-card-title-spacer-y);color:var(--bs-card-title-color)}.card-subtitle{margin-top:calc(-.5 * var(--bs-card-title-spacer-y));margin-bottom:0;color:var(--bs-card-subtitle-color)}.card-text:last-child{margin-bottom:0}.card-link+.card-link{margin-left:var(--bs-card-spacer-x)}.card-header{padding:var(--bs-card-cap-padding-y) var(--bs-card-cap-padding-x);margin-bottom:0;color:var(--bs-card-cap-color);background-color:var(--bs-card-cap-bg);border-bottom:var(--bs-card-border-width) solid var(--bs-card-border-color)}.card-header:first-child{border-radius:var(--bs-card-inner-border-radius) var(--bs-card-inner-border-radius) 0 0}.card-footer{padding:var(--bs-card-cap-padding-y) var(--bs-card-cap-padding-x);color:var(--bs-card-cap-color);background-color:var(--bs-card-cap-bg);border-top:var(--bs-card-border-width) solid var(--bs-card-border-color)}.card-footer:last-child{border-radius:0 0 var(--bs-card-inner-border-radius) var(--bs-card-inner-border-radius)}.card-header-tabs{margin-right:calc(-.5 * var(--bs-card-cap-padding-x));margin-bottom:calc(-1 * var(--bs-card-cap-padding-y));margin-left:calc(-.5 * var(--bs-card-cap-padding-x));border-bottom:0}.card-header-tabs .nav-link.active{background-color:var(--bs-card-bg);border-bottom-color:var(--bs-card-bg)}.card-header-pills{margin-right:calc(-.5 * var(--bs-card-cap-padding-x));margin-left:calc(-.5 * var(--bs-card-cap-padding-x))}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:var(--bs-card-img-overlay-padding);border-radius:var(--bs-card-inner-border-radius)}.card-img,.card-img-bottom,.card-img-top{width:100%}.card-img,.card-img-top{border-top-left-radius:var(--bs-card-inner-border-radius);border-top-right-radius:var(--bs-card-inner-border-radius)}.card-img,.card-img-bottom{border-bottom-right-radius:var(--bs-card-inner-border-radius);border-bottom-left-radius:var(--bs-card-inner-border-radius)}.card-group>.card{margin-bottom:var(--bs-card-group-margin)}@media (min-width:576px){.card-group{display:flex;flex-flow:row wrap}.card-group>.card{flex:1 0 0;margin-bottom:0}.card-group>.card+.card{margin-left:0;border-left:0}.card-group>.card:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.card-group>.card:not(:last-child)>.card-header,.card-group>.card:not(:last-child)>.card-img-top{border-top-right-radius:0}.card-group>.card:not(:last-child)>.card-footer,.card-group>.card:not(:last-child)>.card-img-bottom{border-bottom-right-radius:0}.card-group>.card:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.card-group>.card:not(:first-child)>.card-header,.card-group>.card:not(:first-child)>.card-img-top{border-top-left-radius:0}.card-group>.card:not(:first-child)>.card-footer,.card-group>.card:not(:first-child)>.card-img-bottom{border-bottom-left-radius:0}}.accordion{--bs-accordion-color:var(--bs-body-color);--bs-accordion-bg:var(--bs-body-bg);--bs-accordion-transition:color 0.15s ease-in-out,background-color 0.15s ease-in-out,border-color 0.15s ease-in-out,box-shadow 0.15s ease-in-out,border-radius 0.15s ease;--bs-accordion-border-color:var(--bs-border-color);--bs-accordion-border-width:var(--bs-border-width);--bs-accordion-border-radius:var(--bs-border-radius);--bs-accordion-inner-border-radius:calc(var(--bs-border-radius) - (var(--bs-border-width)));--bs-accordion-btn-padding-x:1.25rem;--bs-accordion-btn-padding-y:1rem;--bs-accordion-btn-color:var(--bs-body-color);--bs-accordion-btn-bg:var(--bs-accordion-bg);--bs-accordion-btn-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='none' stroke='%23212529' stroke-linecap='round' stroke-linejoin='round'%3e%3cpath d='m2 5 6 6 6-6'/%3e%3c/svg%3e");--bs-accordion-btn-icon-width:1.25rem;--bs-accordion-btn-icon-transform:rotate(-180deg);--bs-accordion-btn-icon-transition:transform 0.2s ease-in-out;--bs-accordion-btn-active-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='none' stroke='%23052c65' stroke-linecap='round' stroke-linejoin='round'%3e%3cpath d='m2 5 6 6 6-6'/%3e%3c/svg%3e");--bs-accordion-btn-focus-box-shadow:0 0 0 0.25rem rgba(13, 110, 253, 0.25);--bs-accordion-body-padding-x:1.25rem;--bs-accordion-body-padding-y:1rem;--bs-accordion-active-color:var(--bs-primary-text-emphasis);--bs-accordion-active-bg:var(--bs-primary-bg-subtle)}.accordion-button{position:relative;display:flex;align-items:center;width:100%;padding:var(--bs-accordion-btn-padding-y) var(--bs-accordion-btn-padding-x);font-size:1rem;color:var(--bs-accordion-btn-color);text-align:left;background-color:var(--bs-accordion-btn-bg);border:0;border-radius:0;overflow-anchor:none;transition:var(--bs-accordion-transition)}@media (prefers-reduced-motion:reduce){.accordion-button{transition:none}}.accordion-button:not(.collapsed){color:var(--bs-accordion-active-color);background-color:var(--bs-accordion-active-bg);box-shadow:inset 0 calc(-1 * var(--bs-accordion-border-width)) 0 var(--bs-accordion-border-color)}.accordion-button:not(.collapsed)::after{background-image:var(--bs-accordion-btn-active-icon);transform:var(--bs-accordion-btn-icon-transform)}.accordion-button::after{flex-shrink:0;width:var(--bs-accordion-btn-icon-width);height:var(--bs-accordion-btn-icon-width);margin-left:auto;content:"";background-image:var(--bs-accordion-btn-icon);background-repeat:no-repeat;background-size:var(--bs-accordion-btn-icon-width);transition:var(--bs-accordion-btn-icon-transition)}@media (prefers-reduced-motion:reduce){.accordion-button::after{transition:none}}.accordion-button:hover{z-index:2}.accordion-button:focus{z-index:3;outline:0;box-shadow:var(--bs-accordion-btn-focus-box-shadow)}.accordion-header{margin-bottom:0}.accordion-item{color:var(--bs-accordion-color);background-color:var(--bs-accordion-bg);border:var(--bs-accordion-border-width) solid var(--bs-accordion-border-color)}.accordion-item:first-of-type{border-top-left-radius:var(--bs-accordion-border-radius);border-top-right-radius:var(--bs-accordion-border-radius)}.accordion-item:first-of-type>.accordion-header .accordion-button{border-top-left-radius:var(--bs-accordion-inner-border-radius);border-top-right-radius:var(--bs-accordion-inner-border-radius)}.accordion-item:not(:first-of-type){border-top:0}.accordion-item:last-of-type{border-bottom-right-radius:var(--bs-accordion-border-radius);border-bottom-left-radius:var(--bs-accordion-border-radius)}.accordion-item:last-of-type>.accordion-header .accordion-button.collapsed{border-bottom-right-radius:var(--bs-accordion-inner-border-radius);border-bottom-left-radius:var(--bs-accordion-inner-border-radius)}.accordion-item:last-of-type>.accordion-collapse{border-bottom-right-radius:var(--bs-accordion-border-radius);border-bottom-left-radius:var(--bs-accordion-border-radius)}.accordion-body{padding:var(--bs-accordion-body-padding-y) var(--bs-accordion-body-padding-x)}.accordion-flush>.accordion-item{border-right:0;border-left:0;border-radius:0}.accordion-flush>.accordion-item:first-child{border-top:0}.accordion-flush>.accordion-item:last-child{border-bottom:0}.accordion-flush>.accordion-item>.accordion-collapse,.accordion-flush>.accordion-item>.accordion-header .accordion-button,.accordion-flush>.accordion-item>.accordion-header .accordion-button.collapsed{border-radius:0}[data-bs-theme=dark] .accordion-button::after{--bs-accordion-btn-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%236ea8fe'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708'/%3e%3c/svg%3e");--bs-accordion-btn-active-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%236ea8fe'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708'/%3e%3c/svg%3e")}.breadcrumb{--bs-breadcrumb-padding-x:0;--bs-breadcrumb-padding-y:0;--bs-breadcrumb-margin-bottom:1rem;--bs-breadcrumb-bg: ;--bs-breadcrumb-border-radius: ;--bs-breadcrumb-divider-color:var(--bs-secondary-color);--bs-breadcrumb-item-padding-x:0.5rem;--bs-breadcrumb-item-active-color:var(--bs-secondary-color);display:flex;flex-wrap:wrap;padding:var(--bs-breadcrumb-padding-y) var(--bs-breadcrumb-padding-x);margin-bottom:var(--bs-breadcrumb-margin-bottom);font-size:var(--bs-breadcrumb-font-size);list-style:none;background-color:var(--bs-breadcrumb-bg);border-radius:var(--bs-breadcrumb-border-radius)}.breadcrumb-item+.breadcrumb-item{padding-left:var(--bs-breadcrumb-item-padding-x)}.breadcrumb-item+.breadcrumb-item::before{float:left;padding-right:var(--bs-breadcrumb-item-padding-x);color:var(--bs-breadcrumb-divider-color);content:var(--bs-breadcrumb-divider, "/")}.breadcrumb-item.active{color:var(--bs-breadcrumb-item-active-color)}.pagination{--bs-pagination-padding-x:0.75rem;--bs-pagination-padding-y:0.375rem;--bs-pagination-font-size:1rem;--bs-pagination-color:var(--bs-link-color);--bs-pagination-bg:var(--bs-body-bg);--bs-pagination-border-width:var(--bs-border-width);--bs-pagination-border-color:var(--bs-border-color);--bs-pagination-border-radius:var(--bs-border-radius);--bs-pagination-hover-color:var(--bs-link-hover-color);--bs-pagination-hover-bg:var(--bs-tertiary-bg);--bs-pagination-hover-border-color:var(--bs-border-color);--bs-pagination-focus-color:var(--bs-link-hover-color);--bs-pagination-focus-bg:var(--bs-secondary-bg);--bs-pagination-focus-box-shadow:0 0 0 0.25rem rgba(13, 110, 253, 0.25);--bs-pagination-active-color:#fff;--bs-pagination-active-bg:#0d6efd;--bs-pagination-active-border-color:#0d6efd;--bs-pagination-disabled-color:var(--bs-secondary-color);--bs-pagination-disabled-bg:var(--bs-secondary-bg);--bs-pagination-disabled-border-color:var(--bs-border-color);display:flex;padding-left:0;list-style:none}.page-link{position:relative;display:block;padding:var(--bs-pagination-padding-y) var(--bs-pagination-padding-x);font-size:var(--bs-pagination-font-size);color:var(--bs-pagination-color);text-decoration:none;background-color:var(--bs-pagination-bg);border:var(--bs-pagination-border-width) solid var(--bs-pagination-border-color);transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.page-link{transition:none}}.page-link:hover{z-index:2;color:var(--bs-pagination-hover-color);background-color:var(--bs-pagination-hover-bg);border-color:var(--bs-pagination-hover-border-color)}.page-link:focus{z-index:3;color:var(--bs-pagination-focus-color);background-color:var(--bs-pagination-focus-bg);outline:0;box-shadow:var(--bs-pagination-focus-box-shadow)}.active>.page-link,.page-link.active{z-index:3;color:var(--bs-pagination-active-color);background-color:var(--bs-pagination-active-bg);border-color:var(--bs-pagination-active-border-color)}.disabled>.page-link,.page-link.disabled{color:var(--bs-pagination-disabled-color);pointer-events:none;background-color:var(--bs-pagination-disabled-bg);border-color:var(--bs-pagination-disabled-border-color)}.page-item:not(:first-child) .page-link{margin-left:calc(-1 * var(--bs-border-width))}.page-item:first-child .page-link{border-top-left-radius:var(--bs-pagination-border-radius);border-bottom-left-radius:var(--bs-pagination-border-radius)}.page-item:last-child .page-link{border-top-right-radius:var(--bs-pagination-border-radius);border-bottom-right-radius:var(--bs-pagination-border-radius)}.pagination-lg{--bs-pagination-padding-x:1.5rem;--bs-pagination-padding-y:0.75rem;--bs-pagination-font-size:1.25rem;--bs-pagination-border-radius:var(--bs-border-radius-lg)}.pagination-sm{--bs-pagination-padding-x:0.5rem;--bs-pagination-padding-y:0.25rem;--bs-pagination-font-size:0.875rem;--bs-pagination-border-radius:var(--bs-border-radius-sm)}.badge{--bs-badge-padding-x:0.65em;--bs-badge-padding-y:0.35em;--bs-badge-font-size:0.75em;--bs-badge-font-weight:700;--bs-badge-color:#fff;--bs-badge-border-radius:var(--bs-border-radius);display:inline-block;padding:var(--bs-badge-padding-y) var(--bs-badge-padding-x);font-size:var(--bs-badge-font-size);font-weight:var(--bs-badge-font-weight);line-height:1;color:var(--bs-badge-color);text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:var(--bs-badge-border-radius)}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.alert{--bs-alert-bg:transparent;--bs-alert-padding-x:1rem;--bs-alert-padding-y:1rem;--bs-alert-margin-bottom:1rem;--bs-alert-color:inherit;--bs-alert-border-color:transparent;--bs-alert-border:var(--bs-border-width) solid var(--bs-alert-border-color);--bs-alert-border-radius:var(--bs-border-radius);--bs-alert-link-color:inherit;position:relative;padding:var(--bs-alert-padding-y) var(--bs-alert-padding-x);margin-bottom:var(--bs-alert-margin-bottom);color:var(--bs-alert-color);background-color:var(--bs-alert-bg);border:var(--bs-alert-border);border-radius:var(--bs-alert-border-radius)}.alert-heading{color:inherit}.alert-link{font-weight:700;color:var(--bs-alert-link-color)}.alert-dismissible{padding-right:3rem}.alert-dismissible .btn-close{position:absolute;top:0;right:0;z-index:2;padding:1.25rem 1rem}.alert-primary{--bs-alert-color:var(--bs-primary-text-emphasis);--bs-alert-bg:var(--bs-primary-bg-subtle);--bs-alert-border-color:var(--bs-primary-border-subtle);--bs-alert-link-color:var(--bs-primary-text-emphasis)}.alert-secondary{--bs-alert-color:var(--bs-secondary-text-emphasis);--bs-alert-bg:var(--bs-secondary-bg-subtle);--bs-alert-border-color:var(--bs-secondary-border-subtle);--bs-alert-link-color:var(--bs-secondary-text-emphasis)}.alert-success{--bs-alert-color:var(--bs-success-text-emphasis);--bs-alert-bg:var(--bs-success-bg-subtle);--bs-alert-border-color:var(--bs-success-border-subtle);--bs-alert-link-color:var(--bs-success-text-emphasis)}.alert-info{--bs-alert-color:var(--bs-info-text-emphasis);--bs-alert-bg:var(--bs-info-bg-subtle);--bs-alert-border-color:var(--bs-info-border-subtle);--bs-alert-link-color:var(--bs-info-text-emphasis)}.alert-warning{--bs-alert-color:var(--bs-warning-text-emphasis);--bs-alert-bg:var(--bs-warning-bg-subtle);--bs-alert-border-color:var(--bs-warning-border-subtle);--bs-alert-link-color:var(--bs-warning-text-emphasis)}.alert-danger{--bs-alert-color:var(--bs-danger-text-emphasis);--bs-alert-bg:var(--bs-danger-bg-subtle);--bs-alert-border-color:var(--bs-danger-border-subtle);--bs-alert-link-color:var(--bs-danger-text-emphasis)}.alert-light{--bs-alert-color:var(--bs-light-text-emphasis);--bs-alert-bg:var(--bs-light-bg-subtle);--bs-alert-border-color:var(--bs-light-border-subtle);--bs-alert-link-color:var(--bs-light-text-emphasis)}.alert-dark{--bs-alert-color:var(--bs-dark-text-emphasis);--bs-alert-bg:var(--bs-dark-bg-subtle);--bs-alert-border-color:var(--bs-dark-border-subtle);--bs-alert-link-color:var(--bs-dark-text-emphasis)}@keyframes progress-bar-stripes{0%{background-position-x:var(--bs-progress-height)}}.progress,.progress-stacked{--bs-progress-height:1rem;--bs-progress-font-size:0.75rem;--bs-progress-bg:var(--bs-secondary-bg);--bs-progress-border-radius:var(--bs-border-radius);--bs-progress-box-shadow:var(--bs-box-shadow-inset);--bs-progress-bar-color:#fff;--bs-progress-bar-bg:#0d6efd;--bs-progress-bar-transition:width 0.6s ease;display:flex;height:var(--bs-progress-height);overflow:hidden;font-size:var(--bs-progress-font-size);background-color:var(--bs-progress-bg);border-radius:var(--bs-progress-border-radius)}.progress-bar{display:flex;flex-direction:column;justify-content:center;overflow:hidden;color:var(--bs-progress-bar-color);text-align:center;white-space:nowrap;background-color:var(--bs-progress-bar-bg);transition:var(--bs-progress-bar-transition)}@media (prefers-reduced-motion:reduce){.progress-bar{transition:none}}.progress-bar-striped{background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-size:var(--bs-progress-height) var(--bs-progress-height)}.progress-stacked>.progress{overflow:visible}.progress-stacked>.progress>.progress-bar{width:100%}.progress-bar-animated{animation:1s linear infinite progress-bar-stripes}@media (prefers-reduced-motion:reduce){.progress-bar-animated{animation:none}}.list-group{--bs-list-group-color:var(--bs-body-color);--bs-list-group-bg:var(--bs-body-bg);--bs-list-group-border-color:var(--bs-border-color);--bs-list-group-border-width:var(--bs-border-width);--bs-list-group-border-radius:var(--bs-border-radius);--bs-list-group-item-padding-x:1rem;--bs-list-group-item-padding-y:0.5rem;--bs-list-group-action-color:var(--bs-secondary-color);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-tertiary-bg);--bs-list-group-action-active-color:var(--bs-body-color);--bs-list-group-action-active-bg:var(--bs-secondary-bg);--bs-list-group-disabled-color:var(--bs-secondary-color);--bs-list-group-disabled-bg:var(--bs-body-bg);--bs-list-group-active-color:#fff;--bs-list-group-active-bg:#0d6efd;--bs-list-group-active-border-color:#0d6efd;display:flex;flex-direction:column;padding-left:0;margin-bottom:0;border-radius:var(--bs-list-group-border-radius)}.list-group-numbered{list-style-type:none;counter-reset:section}.list-group-numbered>.list-group-item::before{content:counters(section, ".") ". ";counter-increment:section}.list-group-item{position:relative;display:block;padding:var(--bs-list-group-item-padding-y) var(--bs-list-group-item-padding-x);color:var(--bs-list-group-color);text-decoration:none;background-color:var(--bs-list-group-bg);border:var(--bs-list-group-border-width) solid var(--bs-list-group-border-color)}.list-group-item:first-child{border-top-left-radius:inherit;border-top-right-radius:inherit}.list-group-item:last-child{border-bottom-right-radius:inherit;border-bottom-left-radius:inherit}.list-group-item.disabled,.list-group-item:disabled{color:var(--bs-list-group-disabled-color);pointer-events:none;background-color:var(--bs-list-group-disabled-bg)}.list-group-item.active{z-index:2;color:var(--bs-list-group-active-color);background-color:var(--bs-list-group-active-bg);border-color:var(--bs-list-group-active-border-color)}.list-group-item+.list-group-item{border-top-width:0}.list-group-item+.list-group-item.active{margin-top:calc(-1 * var(--bs-list-group-border-width));border-top-width:var(--bs-list-group-border-width)}.list-group-item-action{width:100%;color:var(--bs-list-group-action-color);text-align:inherit}.list-group-item-action:not(.active):focus,.list-group-item-action:not(.active):hover{z-index:1;color:var(--bs-list-group-action-hover-color);text-decoration:none;background-color:var(--bs-list-group-action-hover-bg)}.list-group-item-action:not(.active):active{color:var(--bs-list-group-action-active-color);background-color:var(--bs-list-group-action-active-bg)}.list-group-horizontal{flex-direction:row}.list-group-horizontal>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal>.list-group-item.active{margin-top:0}.list-group-horizontal>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}@media (min-width:576px){.list-group-horizontal-sm{flex-direction:row}.list-group-horizontal-sm>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-sm>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-sm>.list-group-item.active{margin-top:0}.list-group-horizontal-sm>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-sm>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:768px){.list-group-horizontal-md{flex-direction:row}.list-group-horizontal-md>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-md>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-md>.list-group-item.active{margin-top:0}.list-group-horizontal-md>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-md>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:992px){.list-group-horizontal-lg{flex-direction:row}.list-group-horizontal-lg>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-lg>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-lg>.list-group-item.active{margin-top:0}.list-group-horizontal-lg>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-lg>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:1200px){.list-group-horizontal-xl{flex-direction:row}.list-group-horizontal-xl>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-xl>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-xl>.list-group-item.active{margin-top:0}.list-group-horizontal-xl>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-xl>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:1400px){.list-group-horizontal-xxl{flex-direction:row}.list-group-horizontal-xxl>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-xxl>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-xxl>.list-group-item.active{margin-top:0}.list-group-horizontal-xxl>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-xxl>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}.list-group-flush{border-radius:0}.list-group-flush>.list-group-item{border-width:0 0 var(--bs-list-group-border-width)}.list-group-flush>.list-group-item:last-child{border-bottom-width:0}.list-group-item-primary{--bs-list-group-color:var(--bs-primary-text-emphasis);--bs-list-group-bg:var(--bs-primary-bg-subtle);--bs-list-group-border-color:var(--bs-primary-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-primary-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-primary-border-subtle);--bs-list-group-active-color:var(--bs-primary-bg-subtle);--bs-list-group-active-bg:var(--bs-primary-text-emphasis);--bs-list-group-active-border-color:var(--bs-primary-text-emphasis)}.list-group-item-secondary{--bs-list-group-color:var(--bs-secondary-text-emphasis);--bs-list-group-bg:var(--bs-secondary-bg-subtle);--bs-list-group-border-color:var(--bs-secondary-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-secondary-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-secondary-border-subtle);--bs-list-group-active-color:var(--bs-secondary-bg-subtle);--bs-list-group-active-bg:var(--bs-secondary-text-emphasis);--bs-list-group-active-border-color:var(--bs-secondary-text-emphasis)}.list-group-item-success{--bs-list-group-color:var(--bs-success-text-emphasis);--bs-list-group-bg:var(--bs-success-bg-subtle);--bs-list-group-border-color:var(--bs-success-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-success-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-success-border-subtle);--bs-list-group-active-color:var(--bs-success-bg-subtle);--bs-list-group-active-bg:var(--bs-success-text-emphasis);--bs-list-group-active-border-color:var(--bs-success-text-emphasis)}.list-group-item-info{--bs-list-group-color:var(--bs-info-text-emphasis);--bs-list-group-bg:var(--bs-info-bg-subtle);--bs-list-group-border-color:var(--bs-info-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-info-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-info-border-subtle);--bs-list-group-active-color:var(--bs-info-bg-subtle);--bs-list-group-active-bg:var(--bs-info-text-emphasis);--bs-list-group-active-border-color:var(--bs-info-text-emphasis)}.list-group-item-warning{--bs-list-group-color:var(--bs-warning-text-emphasis);--bs-list-group-bg:var(--bs-warning-bg-subtle);--bs-list-group-border-color:var(--bs-warning-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-warning-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-warning-border-subtle);--bs-list-group-active-color:var(--bs-warning-bg-subtle);--bs-list-group-active-bg:var(--bs-warning-text-emphasis);--bs-list-group-active-border-color:var(--bs-warning-text-emphasis)}.list-group-item-danger{--bs-list-group-color:var(--bs-danger-text-emphasis);--bs-list-group-bg:var(--bs-danger-bg-subtle);--bs-list-group-border-color:var(--bs-danger-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-danger-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-danger-border-subtle);--bs-list-group-active-color:var(--bs-danger-bg-subtle);--bs-list-group-active-bg:var(--bs-danger-text-emphasis);--bs-list-group-active-border-color:var(--bs-danger-text-emphasis)}.list-group-item-light{--bs-list-group-color:var(--bs-light-text-emphasis);--bs-list-group-bg:var(--bs-light-bg-subtle);--bs-list-group-border-color:var(--bs-light-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-light-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-light-border-subtle);--bs-list-group-active-color:var(--bs-light-bg-subtle);--bs-list-group-active-bg:var(--bs-light-text-emphasis);--bs-list-group-active-border-color:var(--bs-light-text-emphasis)}.list-group-item-dark{--bs-list-group-color:var(--bs-dark-text-emphasis);--bs-list-group-bg:var(--bs-dark-bg-subtle);--bs-list-group-border-color:var(--bs-dark-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-dark-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-dark-border-subtle);--bs-list-group-active-color:var(--bs-dark-bg-subtle);--bs-list-group-active-bg:var(--bs-dark-text-emphasis);--bs-list-group-active-border-color:var(--bs-dark-text-emphasis)}.btn-close{--bs-btn-close-color:#000;--bs-btn-close-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23000'%3e%3cpath d='M.293.293a1 1 0 0 1 1.414 0L8 6.586 14.293.293a1 1 0 1 1 1.414 1.414L9.414 8l6.293 6.293a1 1 0 0 1-1.414 1.414L8 9.414l-6.293 6.293a1 1 0 0 1-1.414-1.414L6.586 8 .293 1.707a1 1 0 0 1 0-1.414'/%3e%3c/svg%3e");--bs-btn-close-opacity:0.5;--bs-btn-close-hover-opacity:0.75;--bs-btn-close-focus-shadow:0 0 0 0.25rem rgba(13, 110, 253, 0.25);--bs-btn-close-focus-opacity:1;--bs-btn-close-disabled-opacity:0.25;box-sizing:content-box;width:1em;height:1em;padding:.25em .25em;color:var(--bs-btn-close-color);background:transparent var(--bs-btn-close-bg) center/1em auto no-repeat;filter:var(--bs-btn-close-filter);border:0;border-radius:.375rem;opacity:var(--bs-btn-close-opacity)}.btn-close:hover{color:var(--bs-btn-close-color);text-decoration:none;opacity:var(--bs-btn-close-hover-opacity)}.btn-close:focus{outline:0;box-shadow:var(--bs-btn-close-focus-shadow);opacity:var(--bs-btn-close-focus-opacity)}.btn-close.disabled,.btn-close:disabled{pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none;opacity:var(--bs-btn-close-disabled-opacity)}.btn-close-white{--bs-btn-close-filter:invert(1) grayscale(100%) brightness(200%)}:root,[data-bs-theme=light]{--bs-btn-close-filter: }[data-bs-theme=dark]{--bs-btn-close-filter:invert(1) grayscale(100%) brightness(200%)}.toast{--bs-toast-zindex:1090;--bs-toast-padding-x:0.75rem;--bs-toast-padding-y:0.5rem;--bs-toast-spacing:1.5rem;--bs-toast-max-width:350px;--bs-toast-font-size:0.875rem;--bs-toast-color: ;--bs-toast-bg:rgba(var(--bs-body-bg-rgb), 0.85);--bs-toast-border-width:var(--bs-border-width);--bs-toast-border-color:var(--bs-border-color-translucent);--bs-toast-border-radius:var(--bs-border-radius);--bs-toast-box-shadow:var(--bs-box-shadow);--bs-toast-header-color:var(--bs-secondary-color);--bs-toast-header-bg:rgba(var(--bs-body-bg-rgb), 0.85);--bs-toast-header-border-color:var(--bs-border-color-translucent);width:var(--bs-toast-max-width);max-width:100%;font-size:var(--bs-toast-font-size);color:var(--bs-toast-color);pointer-events:auto;background-color:var(--bs-toast-bg);background-clip:padding-box;border:var(--bs-toast-border-width) solid var(--bs-toast-border-color);box-shadow:var(--bs-toast-box-shadow);border-radius:var(--bs-toast-border-radius)}.toast.showing{opacity:0}.toast:not(.show){display:none}.toast-container{--bs-toast-zindex:1090;position:absolute;z-index:var(--bs-toast-zindex);width:-webkit-max-content;width:-moz-max-content;width:max-content;max-width:100%;pointer-events:none}.toast-container>:not(:last-child){margin-bottom:var(--bs-toast-spacing)}.toast-header{display:flex;align-items:center;padding:var(--bs-toast-padding-y) var(--bs-toast-padding-x);color:var(--bs-toast-header-color);background-color:var(--bs-toast-header-bg);background-clip:padding-box;border-bottom:var(--bs-toast-border-width) solid var(--bs-toast-header-border-color);border-top-left-radius:calc(var(--bs-toast-border-radius) - var(--bs-toast-border-width));border-top-right-radius:calc(var(--bs-toast-border-radius) - var(--bs-toast-border-width))}.toast-header .btn-close{margin-right:calc(-.5 * var(--bs-toast-padding-x));margin-left:var(--bs-toast-padding-x)}.toast-body{padding:var(--bs-toast-padding-x);word-wrap:break-word}.modal{--bs-modal-zindex:1055;--bs-modal-width:500px;--bs-modal-padding:1rem;--bs-modal-margin:0.5rem;--bs-modal-color:var(--bs-body-color);--bs-modal-bg:var(--bs-body-bg);--bs-modal-border-color:var(--bs-border-color-translucent);--bs-modal-border-width:var(--bs-border-width);--bs-modal-border-radius:var(--bs-border-radius-lg);--bs-modal-box-shadow:var(--bs-box-shadow-sm);--bs-modal-inner-border-radius:calc(var(--bs-border-radius-lg) - (var(--bs-border-width)));--bs-modal-header-padding-x:1rem;--bs-modal-header-padding-y:1rem;--bs-modal-header-padding:1rem 1rem;--bs-modal-header-border-color:var(--bs-border-color);--bs-modal-header-border-width:var(--bs-border-width);--bs-modal-title-line-height:1.5;--bs-modal-footer-gap:0.5rem;--bs-modal-footer-bg: ;--bs-modal-footer-border-color:var(--bs-border-color);--bs-modal-footer-border-width:var(--bs-border-width);position:fixed;top:0;left:0;z-index:var(--bs-modal-zindex);display:none;width:100%;height:100%;overflow-x:hidden;overflow-y:auto;outline:0}.modal-dialog{position:relative;width:auto;margin:var(--bs-modal-margin);pointer-events:none}.modal.fade .modal-dialog{transform:translate(0,-50px);transition:transform .3s ease-out}@media (prefers-reduced-motion:reduce){.modal.fade .modal-dialog{transition:none}}.modal.show .modal-dialog{transform:none}.modal.modal-static .modal-dialog{transform:scale(1.02)}.modal-dialog-scrollable{height:calc(100% - var(--bs-modal-margin) * 2)}.modal-dialog-scrollable .modal-content{max-height:100%;overflow:hidden}.modal-dialog-scrollable .modal-body{overflow-y:auto}.modal-dialog-centered{display:flex;align-items:center;min-height:calc(100% - var(--bs-modal-margin) * 2)}.modal-content{position:relative;display:flex;flex-direction:column;width:100%;color:var(--bs-modal-color);pointer-events:auto;background-color:var(--bs-modal-bg);background-clip:padding-box;border:var(--bs-modal-border-width) solid var(--bs-modal-border-color);border-radius:var(--bs-modal-border-radius);outline:0}.modal-backdrop{--bs-backdrop-zindex:1050;--bs-backdrop-bg:#000;--bs-backdrop-opacity:0.5;position:fixed;top:0;left:0;z-index:var(--bs-backdrop-zindex);width:100vw;height:100vh;background-color:var(--bs-backdrop-bg)}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:var(--bs-backdrop-opacity)}.modal-header{display:flex;flex-shrink:0;align-items:center;padding:var(--bs-modal-header-padding);border-bottom:var(--bs-modal-header-border-width) solid var(--bs-modal-header-border-color);border-top-left-radius:var(--bs-modal-inner-border-radius);border-top-right-radius:var(--bs-modal-inner-border-radius)}.modal-header .btn-close{padding:calc(var(--bs-modal-header-padding-y) * .5) calc(var(--bs-modal-header-padding-x) * .5);margin-top:calc(-.5 * var(--bs-modal-header-padding-y));margin-right:calc(-.5 * var(--bs-modal-header-padding-x));margin-bottom:calc(-.5 * var(--bs-modal-header-padding-y));margin-left:auto}.modal-title{margin-bottom:0;line-height:var(--bs-modal-title-line-height)}.modal-body{position:relative;flex:1 1 auto;padding:var(--bs-modal-padding)}.modal-footer{display:flex;flex-shrink:0;flex-wrap:wrap;align-items:center;justify-content:flex-end;padding:calc(var(--bs-modal-padding) - var(--bs-modal-footer-gap) * .5);background-color:var(--bs-modal-footer-bg);border-top:var(--bs-modal-footer-border-width) solid var(--bs-modal-footer-border-color);border-bottom-right-radius:var(--bs-modal-inner-border-radius);border-bottom-left-radius:var(--bs-modal-inner-border-radius)}.modal-footer>*{margin:calc(var(--bs-modal-footer-gap) * .5)}@media (min-width:576px){.modal{--bs-modal-margin:1.75rem;--bs-modal-box-shadow:var(--bs-box-shadow)}.modal-dialog{max-width:var(--bs-modal-width);margin-right:auto;margin-left:auto}.modal-sm{--bs-modal-width:300px}}@media (min-width:992px){.modal-lg,.modal-xl{--bs-modal-width:800px}}@media (min-width:1200px){.modal-xl{--bs-modal-width:1140px}}.modal-fullscreen{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen .modal-footer,.modal-fullscreen .modal-header{border-radius:0}.modal-fullscreen .modal-body{overflow-y:auto}@media (max-width:575.98px){.modal-fullscreen-sm-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-sm-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-sm-down .modal-footer,.modal-fullscreen-sm-down .modal-header{border-radius:0}.modal-fullscreen-sm-down .modal-body{overflow-y:auto}}@media (max-width:767.98px){.modal-fullscreen-md-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-md-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-md-down .modal-footer,.modal-fullscreen-md-down .modal-header{border-radius:0}.modal-fullscreen-md-down .modal-body{overflow-y:auto}}@media (max-width:991.98px){.modal-fullscreen-lg-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-lg-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-lg-down .modal-footer,.modal-fullscreen-lg-down .modal-header{border-radius:0}.modal-fullscreen-lg-down .modal-body{overflow-y:auto}}@media (max-width:1199.98px){.modal-fullscreen-xl-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-xl-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-xl-down .modal-footer,.modal-fullscreen-xl-down .modal-header{border-radius:0}.modal-fullscreen-xl-down .modal-body{overflow-y:auto}}@media (max-width:1399.98px){.modal-fullscreen-xxl-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-xxl-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-xxl-down .modal-footer,.modal-fullscreen-xxl-down .modal-header{border-radius:0}.modal-fullscreen-xxl-down .modal-body{overflow-y:auto}}.tooltip{--bs-tooltip-zindex:1080;--bs-tooltip-max-width:200px;--bs-tooltip-padding-x:0.5rem;--bs-tooltip-padding-y:0.25rem;--bs-tooltip-margin: ;--bs-tooltip-font-size:0.875rem;--bs-tooltip-color:var(--bs-body-bg);--bs-tooltip-bg:var(--bs-emphasis-color);--bs-tooltip-border-radius:var(--bs-border-radius);--bs-tooltip-opacity:0.9;--bs-tooltip-arrow-width:0.8rem;--bs-tooltip-arrow-height:0.4rem;z-index:var(--bs-tooltip-zindex);display:block;margin:var(--bs-tooltip-margin);font-family:var(--bs-font-sans-serif);font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;white-space:normal;word-spacing:normal;line-break:auto;font-size:var(--bs-tooltip-font-size);word-wrap:break-word;opacity:0}.tooltip.show{opacity:var(--bs-tooltip-opacity)}.tooltip .tooltip-arrow{display:block;width:var(--bs-tooltip-arrow-width);height:var(--bs-tooltip-arrow-height)}.tooltip .tooltip-arrow::before{position:absolute;content:"";border-color:transparent;border-style:solid}.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow,.bs-tooltip-top .tooltip-arrow{bottom:calc(-1 * var(--bs-tooltip-arrow-height))}.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow::before,.bs-tooltip-top .tooltip-arrow::before{top:-1px;border-width:var(--bs-tooltip-arrow-height) calc(var(--bs-tooltip-arrow-width) * .5) 0;border-top-color:var(--bs-tooltip-bg)}.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow,.bs-tooltip-end .tooltip-arrow{left:calc(-1 * var(--bs-tooltip-arrow-height));width:var(--bs-tooltip-arrow-height);height:var(--bs-tooltip-arrow-width)}.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow::before,.bs-tooltip-end .tooltip-arrow::before{right:-1px;border-width:calc(var(--bs-tooltip-arrow-width) * .5) var(--bs-tooltip-arrow-height) calc(var(--bs-tooltip-arrow-width) * .5) 0;border-right-color:var(--bs-tooltip-bg)}.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow,.bs-tooltip-bottom .tooltip-arrow{top:calc(-1 * var(--bs-tooltip-arrow-height))}.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow::before,.bs-tooltip-bottom .tooltip-arrow::before{bottom:-1px;border-width:0 calc(var(--bs-tooltip-arrow-width) * .5) var(--bs-tooltip-arrow-height);border-bottom-color:var(--bs-tooltip-bg)}.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow,.bs-tooltip-start .tooltip-arrow{right:calc(-1 * var(--bs-tooltip-arrow-height));width:var(--bs-tooltip-arrow-height);height:var(--bs-tooltip-arrow-width)}.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow::before,.bs-tooltip-start .tooltip-arrow::before{left:-1px;border-width:calc(var(--bs-tooltip-arrow-width) * .5) 0 calc(var(--bs-tooltip-arrow-width) * .5) var(--bs-tooltip-arrow-height);border-left-color:var(--bs-tooltip-bg)}.tooltip-inner{max-width:var(--bs-tooltip-max-width);padding:var(--bs-tooltip-padding-y) var(--bs-tooltip-padding-x);color:var(--bs-tooltip-color);text-align:center;background-color:var(--bs-tooltip-bg);border-radius:var(--bs-tooltip-border-radius)}.popover{--bs-popover-zindex:1070;--bs-popover-max-width:276px;--bs-popover-font-size:0.875rem;--bs-popover-bg:var(--bs-body-bg);--bs-popover-border-width:var(--bs-border-width);--bs-popover-border-color:var(--bs-border-color-translucent);--bs-popover-border-radius:var(--bs-border-radius-lg);--bs-popover-inner-border-radius:calc(var(--bs-border-radius-lg) - var(--bs-border-width));--bs-popover-box-shadow:var(--bs-box-shadow);--bs-popover-header-padding-x:1rem;--bs-popover-header-padding-y:0.5rem;--bs-popover-header-font-size:1rem;--bs-popover-header-color:inherit;--bs-popover-header-bg:var(--bs-secondary-bg);--bs-popover-body-padding-x:1rem;--bs-popover-body-padding-y:1rem;--bs-popover-body-color:var(--bs-body-color);--bs-popover-arrow-width:1rem;--bs-popover-arrow-height:0.5rem;--bs-popover-arrow-border:var(--bs-popover-border-color);z-index:var(--bs-popover-zindex);display:block;max-width:var(--bs-popover-max-width);font-family:var(--bs-font-sans-serif);font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;white-space:normal;word-spacing:normal;line-break:auto;font-size:var(--bs-popover-font-size);word-wrap:break-word;background-color:var(--bs-popover-bg);background-clip:padding-box;border:var(--bs-popover-border-width) solid var(--bs-popover-border-color);border-radius:var(--bs-popover-border-radius)}.popover .popover-arrow{display:block;width:var(--bs-popover-arrow-width);height:var(--bs-popover-arrow-height)}.popover .popover-arrow::after,.popover .popover-arrow::before{position:absolute;display:block;content:"";border-color:transparent;border-style:solid;border-width:0}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow,.bs-popover-top>.popover-arrow{bottom:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width))}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::before,.bs-popover-top>.popover-arrow::after,.bs-popover-top>.popover-arrow::before{border-width:var(--bs-popover-arrow-height) calc(var(--bs-popover-arrow-width) * .5) 0}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::before,.bs-popover-top>.popover-arrow::before{bottom:0;border-top-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::after,.bs-popover-top>.popover-arrow::after{bottom:var(--bs-popover-border-width);border-top-color:var(--bs-popover-bg)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow,.bs-popover-end>.popover-arrow{left:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width));width:var(--bs-popover-arrow-height);height:var(--bs-popover-arrow-width)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::before,.bs-popover-end>.popover-arrow::after,.bs-popover-end>.popover-arrow::before{border-width:calc(var(--bs-popover-arrow-width) * .5) var(--bs-popover-arrow-height) calc(var(--bs-popover-arrow-width) * .5) 0}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::before,.bs-popover-end>.popover-arrow::before{left:0;border-right-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::after,.bs-popover-end>.popover-arrow::after{left:var(--bs-popover-border-width);border-right-color:var(--bs-popover-bg)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow,.bs-popover-bottom>.popover-arrow{top:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width))}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::before,.bs-popover-bottom>.popover-arrow::after,.bs-popover-bottom>.popover-arrow::before{border-width:0 calc(var(--bs-popover-arrow-width) * .5) var(--bs-popover-arrow-height)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::before,.bs-popover-bottom>.popover-arrow::before{top:0;border-bottom-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::after,.bs-popover-bottom>.popover-arrow::after{top:var(--bs-popover-border-width);border-bottom-color:var(--bs-popover-bg)}.bs-popover-auto[data-popper-placement^=bottom] .popover-header::before,.bs-popover-bottom .popover-header::before{position:absolute;top:0;left:50%;display:block;width:var(--bs-popover-arrow-width);margin-left:calc(-.5 * var(--bs-popover-arrow-width));content:"";border-bottom:var(--bs-popover-border-width) solid var(--bs-popover-header-bg)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow,.bs-popover-start>.popover-arrow{right:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width));width:var(--bs-popover-arrow-height);height:var(--bs-popover-arrow-width)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::before,.bs-popover-start>.popover-arrow::after,.bs-popover-start>.popover-arrow::before{border-width:calc(var(--bs-popover-arrow-width) * .5) 0 calc(var(--bs-popover-arrow-width) * .5) var(--bs-popover-arrow-height)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::before,.bs-popover-start>.popover-arrow::before{right:0;border-left-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::after,.bs-popover-start>.popover-arrow::after{right:var(--bs-popover-border-width);border-left-color:var(--bs-popover-bg)}.popover-header{padding:var(--bs-popover-header-padding-y) var(--bs-popover-header-padding-x);margin-bottom:0;font-size:var(--bs-popover-header-font-size);color:var(--bs-popover-header-color);background-color:var(--bs-popover-header-bg);border-bottom:var(--bs-popover-border-width) solid var(--bs-popover-border-color);border-top-left-radius:var(--bs-popover-inner-border-radius);border-top-right-radius:var(--bs-popover-inner-border-radius)}.popover-header:empty{display:none}.popover-body{padding:var(--bs-popover-body-padding-y) var(--bs-popover-body-padding-x);color:var(--bs-popover-body-color)}.carousel{position:relative}.carousel.pointer-event{touch-action:pan-y}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner::after{display:block;clear:both;content:""}.carousel-item{position:relative;display:none;float:left;width:100%;margin-right:-100%;-webkit-backface-visibility:hidden;backface-visibility:hidden;transition:transform .6s ease-in-out}@media (prefers-reduced-motion:reduce){.carousel-item{transition:none}}.carousel-item-next,.carousel-item-prev,.carousel-item.active{display:block}.active.carousel-item-end,.carousel-item-next:not(.carousel-item-start){transform:translateX(100%)}.active.carousel-item-start,.carousel-item-prev:not(.carousel-item-end){transform:translateX(-100%)}.carousel-fade .carousel-item{opacity:0;transition-property:opacity;transform:none}.carousel-fade .carousel-item-next.carousel-item-start,.carousel-fade .carousel-item-prev.carousel-item-end,.carousel-fade .carousel-item.active{z-index:1;opacity:1}.carousel-fade .active.carousel-item-end,.carousel-fade .active.carousel-item-start{z-index:0;opacity:0;transition:opacity 0s .6s}@media (prefers-reduced-motion:reduce){.carousel-fade .active.carousel-item-end,.carousel-fade .active.carousel-item-start{transition:none}}.carousel-control-next,.carousel-control-prev{position:absolute;top:0;bottom:0;z-index:1;display:flex;align-items:center;justify-content:center;width:15%;padding:0;color:#fff;text-align:center;background:0 0;filter:var(--bs-carousel-control-icon-filter);border:0;opacity:.5;transition:opacity .15s ease}@media (prefers-reduced-motion:reduce){.carousel-control-next,.carousel-control-prev{transition:none}}.carousel-control-next:focus,.carousel-control-next:hover,.carousel-control-prev:focus,.carousel-control-prev:hover{color:#fff;text-decoration:none;outline:0;opacity:.9}.carousel-control-prev{left:0}.carousel-control-next{right:0}.carousel-control-next-icon,.carousel-control-prev-icon{display:inline-block;width:2rem;height:2rem;background-repeat:no-repeat;background-position:50%;background-size:100% 100%}.carousel-control-prev-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23fff'%3e%3cpath d='M11.354 1.646a.5.5 0 0 1 0 .708L5.707 8l5.647 5.646a.5.5 0 0 1-.708.708l-6-6a.5.5 0 0 1 0-.708l6-6a.5.5 0 0 1 .708 0'/%3e%3c/svg%3e")}.carousel-control-next-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23fff'%3e%3cpath d='M4.646 1.646a.5.5 0 0 1 .708 0l6 6a.5.5 0 0 1 0 .708l-6 6a.5.5 0 0 1-.708-.708L10.293 8 4.646 2.354a.5.5 0 0 1 0-.708'/%3e%3c/svg%3e")}.carousel-indicators{position:absolute;right:0;bottom:0;left:0;z-index:2;display:flex;justify-content:center;padding:0;margin-right:15%;margin-bottom:1rem;margin-left:15%}.carousel-indicators [data-bs-target]{box-sizing:content-box;flex:0 1 auto;width:30px;height:3px;padding:0;margin-right:3px;margin-left:3px;text-indent:-999px;cursor:pointer;background-color:var(--bs-carousel-indicator-active-bg);background-clip:padding-box;border:0;border-top:10px solid transparent;border-bottom:10px solid transparent;opacity:.5;transition:opacity .6s ease}@media (prefers-reduced-motion:reduce){.carousel-indicators [data-bs-target]{transition:none}}.carousel-indicators .active{opacity:1}.carousel-caption{position:absolute;right:15%;bottom:1.25rem;left:15%;padding-top:1.25rem;padding-bottom:1.25rem;color:var(--bs-carousel-caption-color);text-align:center}.carousel-dark{--bs-carousel-indicator-active-bg:#000;--bs-carousel-caption-color:#000;--bs-carousel-control-icon-filter:invert(1) grayscale(100)}:root,[data-bs-theme=light]{--bs-carousel-indicator-active-bg:#fff;--bs-carousel-caption-color:#fff;--bs-carousel-control-icon-filter: }[data-bs-theme=dark]{--bs-carousel-indicator-active-bg:#000;--bs-carousel-caption-color:#000;--bs-carousel-control-icon-filter:invert(1) grayscale(100)}.spinner-border,.spinner-grow{display:inline-block;width:var(--bs-spinner-width);height:var(--bs-spinner-height);vertical-align:var(--bs-spinner-vertical-align);border-radius:50%;animation:var(--bs-spinner-animation-speed) linear infinite var(--bs-spinner-animation-name)}@keyframes spinner-border{to{transform:rotate(360deg)}}.spinner-border{--bs-spinner-width:2rem;--bs-spinner-height:2rem;--bs-spinner-vertical-align:-0.125em;--bs-spinner-border-width:0.25em;--bs-spinner-animation-speed:0.75s;--bs-spinner-animation-name:spinner-border;border:var(--bs-spinner-border-width) solid currentcolor;border-right-color:transparent}.spinner-border-sm{--bs-spinner-width:1rem;--bs-spinner-height:1rem;--bs-spinner-border-width:0.2em}@keyframes spinner-grow{0%{transform:scale(0)}50%{opacity:1;transform:none}}.spinner-grow{--bs-spinner-width:2rem;--bs-spinner-height:2rem;--bs-spinner-vertical-align:-0.125em;--bs-spinner-animation-speed:0.75s;--bs-spinner-animation-name:spinner-grow;background-color:currentcolor;opacity:0}.spinner-grow-sm{--bs-spinner-width:1rem;--bs-spinner-height:1rem}@media (prefers-reduced-motion:reduce){.spinner-border,.spinner-grow{--bs-spinner-animation-speed:1.5s}}.offcanvas,.offcanvas-lg,.offcanvas-md,.offcanvas-sm,.offcanvas-xl,.offcanvas-xxl{--bs-offcanvas-zindex:1045;--bs-offcanvas-width:400px;--bs-offcanvas-height:30vh;--bs-offcanvas-padding-x:1rem;--bs-offcanvas-padding-y:1rem;--bs-offcanvas-color:var(--bs-body-color);--bs-offcanvas-bg:var(--bs-body-bg);--bs-offcanvas-border-width:var(--bs-border-width);--bs-offcanvas-border-color:var(--bs-border-color-translucent);--bs-offcanvas-box-shadow:var(--bs-box-shadow-sm);--bs-offcanvas-transition:transform 0.3s ease-in-out;--bs-offcanvas-title-line-height:1.5}@media (max-width:575.98px){.offcanvas-sm{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:575.98px) and (prefers-reduced-motion:reduce){.offcanvas-sm{transition:none}}@media (max-width:575.98px){.offcanvas-sm.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-sm.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-sm.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-sm.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-sm.show:not(.hiding),.offcanvas-sm.showing{transform:none}.offcanvas-sm.hiding,.offcanvas-sm.show,.offcanvas-sm.showing{visibility:visible}}@media (min-width:576px){.offcanvas-sm{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-sm .offcanvas-header{display:none}.offcanvas-sm .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:767.98px){.offcanvas-md{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:767.98px) and (prefers-reduced-motion:reduce){.offcanvas-md{transition:none}}@media (max-width:767.98px){.offcanvas-md.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-md.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-md.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-md.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-md.show:not(.hiding),.offcanvas-md.showing{transform:none}.offcanvas-md.hiding,.offcanvas-md.show,.offcanvas-md.showing{visibility:visible}}@media (min-width:768px){.offcanvas-md{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-md .offcanvas-header{display:none}.offcanvas-md .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:991.98px){.offcanvas-lg{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:991.98px) and (prefers-reduced-motion:reduce){.offcanvas-lg{transition:none}}@media (max-width:991.98px){.offcanvas-lg.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-lg.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-lg.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-lg.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-lg.show:not(.hiding),.offcanvas-lg.showing{transform:none}.offcanvas-lg.hiding,.offcanvas-lg.show,.offcanvas-lg.showing{visibility:visible}}@media (min-width:992px){.offcanvas-lg{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-lg .offcanvas-header{display:none}.offcanvas-lg .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:1199.98px){.offcanvas-xl{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:1199.98px) and (prefers-reduced-motion:reduce){.offcanvas-xl{transition:none}}@media (max-width:1199.98px){.offcanvas-xl.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-xl.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-xl.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-xl.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-xl.show:not(.hiding),.offcanvas-xl.showing{transform:none}.offcanvas-xl.hiding,.offcanvas-xl.show,.offcanvas-xl.showing{visibility:visible}}@media (min-width:1200px){.offcanvas-xl{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-xl .offcanvas-header{display:none}.offcanvas-xl .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:1399.98px){.offcanvas-xxl{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:1399.98px) and (prefers-reduced-motion:reduce){.offcanvas-xxl{transition:none}}@media (max-width:1399.98px){.offcanvas-xxl.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-xxl.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-xxl.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-xxl.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-xxl.show:not(.hiding),.offcanvas-xxl.showing{transform:none}.offcanvas-xxl.hiding,.offcanvas-xxl.show,.offcanvas-xxl.showing{visibility:visible}}@media (min-width:1400px){.offcanvas-xxl{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-xxl .offcanvas-header{display:none}.offcanvas-xxl .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}.offcanvas{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}@media (prefers-reduced-motion:reduce){.offcanvas{transition:none}}.offcanvas.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas.show:not(.hiding),.offcanvas.showing{transform:none}.offcanvas.hiding,.offcanvas.show,.offcanvas.showing{visibility:visible}.offcanvas-backdrop{position:fixed;top:0;left:0;z-index:1040;width:100vw;height:100vh;background-color:#000}.offcanvas-backdrop.fade{opacity:0}.offcanvas-backdrop.show{opacity:.5}.offcanvas-header{display:flex;align-items:center;padding:var(--bs-offcanvas-padding-y) var(--bs-offcanvas-padding-x)}.offcanvas-header .btn-close{padding:calc(var(--bs-offcanvas-padding-y) * .5) calc(var(--bs-offcanvas-padding-x) * .5);margin-top:calc(-.5 * var(--bs-offcanvas-padding-y));margin-right:calc(-.5 * var(--bs-offcanvas-padding-x));margin-bottom:calc(-.5 * var(--bs-offcanvas-padding-y));margin-left:auto}.offcanvas-title{margin-bottom:0;line-height:var(--bs-offcanvas-title-line-height)}.offcanvas-body{flex-grow:1;padding:var(--bs-offcanvas-padding-y) var(--bs-offcanvas-padding-x);overflow-y:auto}.placeholder{display:inline-block;min-height:1em;vertical-align:middle;cursor:wait;background-color:currentcolor;opacity:.5}.placeholder.btn::before{display:inline-block;content:""}.placeholder-xs{min-height:.6em}.placeholder-sm{min-height:.8em}.placeholder-lg{min-height:1.2em}.placeholder-glow .placeholder{animation:placeholder-glow 2s ease-in-out infinite}@keyframes placeholder-glow{50%{opacity:.2}}.placeholder-wave{-webkit-mask-image:linear-gradient(130deg,#000 55%,rgba(0,0,0,0.8) 75%,#000 95%);mask-image:linear-gradient(130deg,#000 55%,rgba(0,0,0,0.8) 75%,#000 95%);-webkit-mask-size:200% 100%;mask-size:200% 100%;animation:placeholder-wave 2s linear infinite}@keyframes placeholder-wave{100%{-webkit-mask-position:-200% 0%;mask-position:-200% 0%}}.clearfix::after{display:block;clear:both;content:""}.text-bg-primary{color:#fff!important;background-color:RGBA(var(--bs-primary-rgb),var(--bs-bg-opacity,1))!important}.text-bg-secondary{color:#fff!important;background-color:RGBA(var(--bs-secondary-rgb),var(--bs-bg-opacity,1))!important}.text-bg-success{color:#fff!important;background-color:RGBA(var(--bs-success-rgb),var(--bs-bg-opacity,1))!important}.text-bg-info{color:#000!important;background-color:RGBA(var(--bs-info-rgb),var(--bs-bg-opacity,1))!important}.text-bg-warning{color:#000!important;background-color:RGBA(var(--bs-warning-rgb),var(--bs-bg-opacity,1))!important}.text-bg-danger{color:#fff!important;background-color:RGBA(var(--bs-danger-rgb),var(--bs-bg-opacity,1))!important}.text-bg-light{color:#000!important;background-color:RGBA(var(--bs-light-rgb),var(--bs-bg-opacity,1))!important}.text-bg-dark{color:#fff!important;background-color:RGBA(var(--bs-dark-rgb),var(--bs-bg-opacity,1))!important}.link-primary{color:RGBA(var(--bs-primary-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-primary-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-primary-rgb),var(--bs-link-underline-opacity,1))!important}.link-primary:focus,.link-primary:hover{color:RGBA(10,88,202,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(10,88,202,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(10,88,202,var(--bs-link-underline-opacity,1))!important}.link-secondary{color:RGBA(var(--bs-secondary-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-secondary-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-secondary-rgb),var(--bs-link-underline-opacity,1))!important}.link-secondary:focus,.link-secondary:hover{color:RGBA(86,94,100,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(86,94,100,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(86,94,100,var(--bs-link-underline-opacity,1))!important}.link-success{color:RGBA(var(--bs-success-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-success-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-success-rgb),var(--bs-link-underline-opacity,1))!important}.link-success:focus,.link-success:hover{color:RGBA(20,108,67,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(20,108,67,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(20,108,67,var(--bs-link-underline-opacity,1))!important}.link-info{color:RGBA(var(--bs-info-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-info-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-info-rgb),var(--bs-link-underline-opacity,1))!important}.link-info:focus,.link-info:hover{color:RGBA(61,213,243,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(61,213,243,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(61,213,243,var(--bs-link-underline-opacity,1))!important}.link-warning{color:RGBA(var(--bs-warning-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-warning-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-warning-rgb),var(--bs-link-underline-opacity,1))!important}.link-warning:focus,.link-warning:hover{color:RGBA(255,205,57,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(255,205,57,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(255,205,57,var(--bs-link-underline-opacity,1))!important}.link-danger{color:RGBA(var(--bs-danger-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-danger-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-danger-rgb),var(--bs-link-underline-opacity,1))!important}.link-danger:focus,.link-danger:hover{color:RGBA(176,42,55,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(176,42,55,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(176,42,55,var(--bs-link-underline-opacity,1))!important}.link-light{color:RGBA(var(--bs-light-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-light-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-light-rgb),var(--bs-link-underline-opacity,1))!important}.link-light:focus,.link-light:hover{color:RGBA(249,250,251,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(249,250,251,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(249,250,251,var(--bs-link-underline-opacity,1))!important}.link-dark{color:RGBA(var(--bs-dark-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-dark-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-dark-rgb),var(--bs-link-underline-opacity,1))!important}.link-dark:focus,.link-dark:hover{color:RGBA(26,30,33,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(26,30,33,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(26,30,33,var(--bs-link-underline-opacity,1))!important}.link-body-emphasis{color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-underline-opacity,1))!important}.link-body-emphasis:focus,.link-body-emphasis:hover{color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-opacity,.75))!important;-webkit-text-decoration-color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-underline-opacity,0.75))!important;text-decoration-color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-underline-opacity,0.75))!important}.focus-ring:focus{outline:0;box-shadow:var(--bs-focus-ring-x,0) var(--bs-focus-ring-y,0) var(--bs-focus-ring-blur,0) var(--bs-focus-ring-width) var(--bs-focus-ring-color)}.icon-link{display:inline-flex;gap:.375rem;align-items:center;-webkit-text-decoration-color:rgba(var(--bs-link-color-rgb),var(--bs-link-opacity,0.5));text-decoration-color:rgba(var(--bs-link-color-rgb),var(--bs-link-opacity,0.5));text-underline-offset:0.25em;-webkit-backface-visibility:hidden;backface-visibility:hidden}.icon-link>.bi{flex-shrink:0;width:1em;height:1em;fill:currentcolor;transition:.2s ease-in-out transform}@media (prefers-reduced-motion:reduce){.icon-link>.bi{transition:none}}.icon-link-hover:focus-visible>.bi,.icon-link-hover:hover>.bi{transform:var(--bs-icon-link-transform,translate3d(.25em,0,0))}.ratio{position:relative;width:100%}.ratio::before{display:block;padding-top:var(--bs-aspect-ratio);content:""}.ratio>*{position:absolute;top:0;left:0;width:100%;height:100%}.ratio-1x1{--bs-aspect-ratio:100%}.ratio-4x3{--bs-aspect-ratio:75%}.ratio-16x9{--bs-aspect-ratio:56.25%}.ratio-21x9{--bs-aspect-ratio:42.8571428571%}.fixed-top{position:fixed;top:0;right:0;left:0;z-index:1030}.fixed-bottom{position:fixed;right:0;bottom:0;left:0;z-index:1030}.sticky-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}@media (min-width:576px){.sticky-sm-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-sm-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:768px){.sticky-md-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-md-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:992px){.sticky-lg-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-lg-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:1200px){.sticky-xl-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-xl-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:1400px){.sticky-xxl-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-xxl-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}.hstack{display:flex;flex-direction:row;align-items:center;align-self:stretch}.vstack{display:flex;flex:1 1 auto;flex-direction:column;align-self:stretch}.visually-hidden,.visually-hidden-focusable:not(:focus):not(:focus-within){width:1px!important;height:1px!important;padding:0!important;margin:-1px!important;overflow:hidden!important;clip:rect(0,0,0,0)!important;white-space:nowrap!important;border:0!important}.visually-hidden-focusable:not(:focus):not(:focus-within):not(caption),.visually-hidden:not(caption){position:absolute!important}.visually-hidden *,.visually-hidden-focusable:not(:focus):not(:focus-within) *{overflow:hidden!important}.stretched-link::after{position:absolute;top:0;right:0;bottom:0;left:0;z-index:1;content:""}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.vr{display:inline-block;align-self:stretch;width:var(--bs-border-width);min-height:1em;background-color:currentcolor;opacity:.25}.align-baseline{vertical-align:baseline!important}.align-top{vertical-align:top!important}.align-middle{vertical-align:middle!important}.align-bottom{vertical-align:bottom!important}.align-text-bottom{vertical-align:text-bottom!important}.align-text-top{vertical-align:text-top!important}.float-start{float:left!important}.float-end{float:right!important}.float-none{float:none!important}.object-fit-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-none{-o-object-fit:none!important;object-fit:none!important}.opacity-0{opacity:0!important}.opacity-25{opacity:.25!important}.opacity-50{opacity:.5!important}.opacity-75{opacity:.75!important}.opacity-100{opacity:1!important}.overflow-auto{overflow:auto!important}.overflow-hidden{overflow:hidden!important}.overflow-visible{overflow:visible!important}.overflow-scroll{overflow:scroll!important}.overflow-x-auto{overflow-x:auto!important}.overflow-x-hidden{overflow-x:hidden!important}.overflow-x-visible{overflow-x:visible!important}.overflow-x-scroll{overflow-x:scroll!important}.overflow-y-auto{overflow-y:auto!important}.overflow-y-hidden{overflow-y:hidden!important}.overflow-y-visible{overflow-y:visible!important}.overflow-y-scroll{overflow-y:scroll!important}.d-inline{display:inline!important}.d-inline-block{display:inline-block!important}.d-block{display:block!important}.d-grid{display:grid!important}.d-inline-grid{display:inline-grid!important}.d-table{display:table!important}.d-table-row{display:table-row!important}.d-table-cell{display:table-cell!important}.d-flex{display:flex!important}.d-inline-flex{display:inline-flex!important}.d-none{display:none!important}.shadow{box-shadow:var(--bs-box-shadow)!important}.shadow-sm{box-shadow:var(--bs-box-shadow-sm)!important}.shadow-lg{box-shadow:var(--bs-box-shadow-lg)!important}.shadow-none{box-shadow:none!important}.focus-ring-primary{--bs-focus-ring-color:rgba(var(--bs-primary-rgb), var(--bs-focus-ring-opacity))}.focus-ring-secondary{--bs-focus-ring-color:rgba(var(--bs-secondary-rgb), var(--bs-focus-ring-opacity))}.focus-ring-success{--bs-focus-ring-color:rgba(var(--bs-success-rgb), var(--bs-focus-ring-opacity))}.focus-ring-info{--bs-focus-ring-color:rgba(var(--bs-info-rgb), var(--bs-focus-ring-opacity))}.focus-ring-warning{--bs-focus-ring-color:rgba(var(--bs-warning-rgb), var(--bs-focus-ring-opacity))}.focus-ring-danger{--bs-focus-ring-color:rgba(var(--bs-danger-rgb), var(--bs-focus-ring-opacity))}.focus-ring-light{--bs-focus-ring-color:rgba(var(--bs-light-rgb), var(--bs-focus-ring-opacity))}.focus-ring-dark{--bs-focus-ring-color:rgba(var(--bs-dark-rgb), var(--bs-focus-ring-opacity))}.position-static{position:static!important}.position-relative{position:relative!important}.position-absolute{position:absolute!important}.position-fixed{position:fixed!important}.position-sticky{position:-webkit-sticky!important;position:sticky!important}.top-0{top:0!important}.top-50{top:50%!important}.top-100{top:100%!important}.bottom-0{bottom:0!important}.bottom-50{bottom:50%!important}.bottom-100{bottom:100%!important}.start-0{left:0!important}.start-50{left:50%!important}.start-100{left:100%!important}.end-0{right:0!important}.end-50{right:50%!important}.end-100{right:100%!important}.translate-middle{transform:translate(-50%,-50%)!important}.translate-middle-x{transform:translateX(-50%)!important}.translate-middle-y{transform:translateY(-50%)!important}.border{border:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-0{border:0!important}.border-top{border-top:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-top-0{border-top:0!important}.border-end{border-right:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-end-0{border-right:0!important}.border-bottom{border-bottom:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-bottom-0{border-bottom:0!important}.border-start{border-left:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-start-0{border-left:0!important}.border-primary{--bs-border-opacity:1;border-color:rgba(var(--bs-primary-rgb),var(--bs-border-opacity))!important}.border-secondary{--bs-border-opacity:1;border-color:rgba(var(--bs-secondary-rgb),var(--bs-border-opacity))!important}.border-success{--bs-border-opacity:1;border-color:rgba(var(--bs-success-rgb),var(--bs-border-opacity))!important}.border-info{--bs-border-opacity:1;border-color:rgba(var(--bs-info-rgb),var(--bs-border-opacity))!important}.border-warning{--bs-border-opacity:1;border-color:rgba(var(--bs-warning-rgb),var(--bs-border-opacity))!important}.border-danger{--bs-border-opacity:1;border-color:rgba(var(--bs-danger-rgb),var(--bs-border-opacity))!important}.border-light{--bs-border-opacity:1;border-color:rgba(var(--bs-light-rgb),var(--bs-border-opacity))!important}.border-dark{--bs-border-opacity:1;border-color:rgba(var(--bs-dark-rgb),var(--bs-border-opacity))!important}.border-black{--bs-border-opacity:1;border-color:rgba(var(--bs-black-rgb),var(--bs-border-opacity))!important}.border-white{--bs-border-opacity:1;border-color:rgba(var(--bs-white-rgb),var(--bs-border-opacity))!important}.border-primary-subtle{border-color:var(--bs-primary-border-subtle)!important}.border-secondary-subtle{border-color:var(--bs-secondary-border-subtle)!important}.border-success-subtle{border-color:var(--bs-success-border-subtle)!important}.border-info-subtle{border-color:var(--bs-info-border-subtle)!important}.border-warning-subtle{border-color:var(--bs-warning-border-subtle)!important}.border-danger-subtle{border-color:var(--bs-danger-border-subtle)!important}.border-light-subtle{border-color:var(--bs-light-border-subtle)!important}.border-dark-subtle{border-color:var(--bs-dark-border-subtle)!important}.border-1{border-width:1px!important}.border-2{border-width:2px!important}.border-3{border-width:3px!important}.border-4{border-width:4px!important}.border-5{border-width:5px!important}.border-opacity-10{--bs-border-opacity:0.1}.border-opacity-25{--bs-border-opacity:0.25}.border-opacity-50{--bs-border-opacity:0.5}.border-opacity-75{--bs-border-opacity:0.75}.border-opacity-100{--bs-border-opacity:1}.w-25{width:25%!important}.w-50{width:50%!important}.w-75{width:75%!important}.w-100{width:100%!important}.w-auto{width:auto!important}.mw-100{max-width:100%!important}.vw-100{width:100vw!important}.min-vw-100{min-width:100vw!important}.h-25{height:25%!important}.h-50{height:50%!important}.h-75{height:75%!important}.h-100{height:100%!important}.h-auto{height:auto!important}.mh-100{max-height:100%!important}.vh-100{height:100vh!important}.min-vh-100{min-height:100vh!important}.flex-fill{flex:1 1 auto!important}.flex-row{flex-direction:row!important}.flex-column{flex-direction:column!important}.flex-row-reverse{flex-direction:row-reverse!important}.flex-column-reverse{flex-direction:column-reverse!important}.flex-grow-0{flex-grow:0!important}.flex-grow-1{flex-grow:1!important}.flex-shrink-0{flex-shrink:0!important}.flex-shrink-1{flex-shrink:1!important}.flex-wrap{flex-wrap:wrap!important}.flex-nowrap{flex-wrap:nowrap!important}.flex-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-start{justify-content:flex-start!important}.justify-content-end{justify-content:flex-end!important}.justify-content-center{justify-content:center!important}.justify-content-between{justify-content:space-between!important}.justify-content-around{justify-content:space-around!important}.justify-content-evenly{justify-content:space-evenly!important}.align-items-start{align-items:flex-start!important}.align-items-end{align-items:flex-end!important}.align-items-center{align-items:center!important}.align-items-baseline{align-items:baseline!important}.align-items-stretch{align-items:stretch!important}.align-content-start{align-content:flex-start!important}.align-content-end{align-content:flex-end!important}.align-content-center{align-content:center!important}.align-content-between{align-content:space-between!important}.align-content-around{align-content:space-around!important}.align-content-stretch{align-content:stretch!important}.align-self-auto{align-self:auto!important}.align-self-start{align-self:flex-start!important}.align-self-end{align-self:flex-end!important}.align-self-center{align-self:center!important}.align-self-baseline{align-self:baseline!important}.align-self-stretch{align-self:stretch!important}.order-first{order:-1!important}.order-0{order:0!important}.order-1{order:1!important}.order-2{order:2!important}.order-3{order:3!important}.order-4{order:4!important}.order-5{order:5!important}.order-last{order:6!important}.m-0{margin:0!important}.m-1{margin:.25rem!important}.m-2{margin:.5rem!important}.m-3{margin:1rem!important}.m-4{margin:1.5rem!important}.m-5{margin:3rem!important}.m-auto{margin:auto!important}.mx-0{margin-right:0!important;margin-left:0!important}.mx-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-3{margin-right:1rem!important;margin-left:1rem!important}.mx-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-5{margin-right:3rem!important;margin-left:3rem!important}.mx-auto{margin-right:auto!important;margin-left:auto!important}.my-0{margin-top:0!important;margin-bottom:0!important}.my-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-0{margin-top:0!important}.mt-1{margin-top:.25rem!important}.mt-2{margin-top:.5rem!important}.mt-3{margin-top:1rem!important}.mt-4{margin-top:1.5rem!important}.mt-5{margin-top:3rem!important}.mt-auto{margin-top:auto!important}.me-0{margin-right:0!important}.me-1{margin-right:.25rem!important}.me-2{margin-right:.5rem!important}.me-3{margin-right:1rem!important}.me-4{margin-right:1.5rem!important}.me-5{margin-right:3rem!important}.me-auto{margin-right:auto!important}.mb-0{margin-bottom:0!important}.mb-1{margin-bottom:.25rem!important}.mb-2{margin-bottom:.5rem!important}.mb-3{margin-bottom:1rem!important}.mb-4{margin-bottom:1.5rem!important}.mb-5{margin-bottom:3rem!important}.mb-auto{margin-bottom:auto!important}.ms-0{margin-left:0!important}.ms-1{margin-left:.25rem!important}.ms-2{margin-left:.5rem!important}.ms-3{margin-left:1rem!important}.ms-4{margin-left:1.5rem!important}.ms-5{margin-left:3rem!important}.ms-auto{margin-left:auto!important}.p-0{padding:0!important}.p-1{padding:.25rem!important}.p-2{padding:.5rem!important}.p-3{padding:1rem!important}.p-4{padding:1.5rem!important}.p-5{padding:3rem!important}.px-0{padding-right:0!important;padding-left:0!important}.px-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-3{padding-right:1rem!important;padding-left:1rem!important}.px-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-5{padding-right:3rem!important;padding-left:3rem!important}.py-0{padding-top:0!important;padding-bottom:0!important}.py-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-0{padding-top:0!important}.pt-1{padding-top:.25rem!important}.pt-2{padding-top:.5rem!important}.pt-3{padding-top:1rem!important}.pt-4{padding-top:1.5rem!important}.pt-5{padding-top:3rem!important}.pe-0{padding-right:0!important}.pe-1{padding-right:.25rem!important}.pe-2{padding-right:.5rem!important}.pe-3{padding-right:1rem!important}.pe-4{padding-right:1.5rem!important}.pe-5{padding-right:3rem!important}.pb-0{padding-bottom:0!important}.pb-1{padding-bottom:.25rem!important}.pb-2{padding-bottom:.5rem!important}.pb-3{padding-bottom:1rem!important}.pb-4{padding-bottom:1.5rem!important}.pb-5{padding-bottom:3rem!important}.ps-0{padding-left:0!important}.ps-1{padding-left:.25rem!important}.ps-2{padding-left:.5rem!important}.ps-3{padding-left:1rem!important}.ps-4{padding-left:1.5rem!important}.ps-5{padding-left:3rem!important}.gap-0{gap:0!important}.gap-1{gap:.25rem!important}.gap-2{gap:.5rem!important}.gap-3{gap:1rem!important}.gap-4{gap:1.5rem!important}.gap-5{gap:3rem!important}.row-gap-0{row-gap:0!important}.row-gap-1{row-gap:.25rem!important}.row-gap-2{row-gap:.5rem!important}.row-gap-3{row-gap:1rem!important}.row-gap-4{row-gap:1.5rem!important}.row-gap-5{row-gap:3rem!important}.column-gap-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.font-monospace{font-family:var(--bs-font-monospace)!important}.fs-1{font-size:calc(1.375rem + 1.5vw)!important}.fs-2{font-size:calc(1.325rem + .9vw)!important}.fs-3{font-size:calc(1.3rem + .6vw)!important}.fs-4{font-size:calc(1.275rem + .3vw)!important}.fs-5{font-size:1.25rem!important}.fs-6{font-size:1rem!important}.fst-italic{font-style:italic!important}.fst-normal{font-style:normal!important}.fw-lighter{font-weight:lighter!important}.fw-light{font-weight:300!important}.fw-normal{font-weight:400!important}.fw-medium{font-weight:500!important}.fw-semibold{font-weight:600!important}.fw-bold{font-weight:700!important}.fw-bolder{font-weight:bolder!important}.lh-1{line-height:1!important}.lh-sm{line-height:1.25!important}.lh-base{line-height:1.5!important}.lh-lg{line-height:2!important}.text-start{text-align:left!important}.text-end{text-align:right!important}.text-center{text-align:center!important}.text-decoration-none{text-decoration:none!important}.text-decoration-underline{text-decoration:underline!important}.text-decoration-line-through{text-decoration:line-through!important}.text-lowercase{text-transform:lowercase!important}.text-uppercase{text-transform:uppercase!important}.text-capitalize{text-transform:capitalize!important}.text-wrap{white-space:normal!important}.text-nowrap{white-space:nowrap!important}.text-break{word-wrap:break-word!important;word-break:break-word!important}.text-primary{--bs-text-opacity:1;color:rgba(var(--bs-primary-rgb),var(--bs-text-opacity))!important}.text-secondary{--bs-text-opacity:1;color:rgba(var(--bs-secondary-rgb),var(--bs-text-opacity))!important}.text-success{--bs-text-opacity:1;color:rgba(var(--bs-success-rgb),var(--bs-text-opacity))!important}.text-info{--bs-text-opacity:1;color:rgba(var(--bs-info-rgb),var(--bs-text-opacity))!important}.text-warning{--bs-text-opacity:1;color:rgba(var(--bs-warning-rgb),var(--bs-text-opacity))!important}.text-danger{--bs-text-opacity:1;color:rgba(var(--bs-danger-rgb),var(--bs-text-opacity))!important}.text-light{--bs-text-opacity:1;color:rgba(var(--bs-light-rgb),var(--bs-text-opacity))!important}.text-dark{--bs-text-opacity:1;color:rgba(var(--bs-dark-rgb),var(--bs-text-opacity))!important}.text-black{--bs-text-opacity:1;color:rgba(var(--bs-black-rgb),var(--bs-text-opacity))!important}.text-white{--bs-text-opacity:1;color:rgba(var(--bs-white-rgb),var(--bs-text-opacity))!important}.text-body{--bs-text-opacity:1;color:rgba(var(--bs-body-color-rgb),var(--bs-text-opacity))!important}.text-muted{--bs-text-opacity:1;color:var(--bs-secondary-color)!important}.text-black-50{--bs-text-opacity:1;color:rgba(0,0,0,.5)!important}.text-white-50{--bs-text-opacity:1;color:rgba(255,255,255,.5)!important}.text-body-secondary{--bs-text-opacity:1;color:var(--bs-secondary-color)!important}.text-body-tertiary{--bs-text-opacity:1;color:var(--bs-tertiary-color)!important}.text-body-emphasis{--bs-text-opacity:1;color:var(--bs-emphasis-color)!important}.text-reset{--bs-text-opacity:1;color:inherit!important}.text-opacity-25{--bs-text-opacity:0.25}.text-opacity-50{--bs-text-opacity:0.5}.text-opacity-75{--bs-text-opacity:0.75}.text-opacity-100{--bs-text-opacity:1}.text-primary-emphasis{color:var(--bs-primary-text-emphasis)!important}.text-secondary-emphasis{color:var(--bs-secondary-text-emphasis)!important}.text-success-emphasis{color:var(--bs-success-text-emphasis)!important}.text-info-emphasis{color:var(--bs-info-text-emphasis)!important}.text-warning-emphasis{color:var(--bs-warning-text-emphasis)!important}.text-danger-emphasis{color:var(--bs-danger-text-emphasis)!important}.text-light-emphasis{color:var(--bs-light-text-emphasis)!important}.text-dark-emphasis{color:var(--bs-dark-text-emphasis)!important}.link-opacity-10{--bs-link-opacity:0.1}.link-opacity-10-hover:hover{--bs-link-opacity:0.1}.link-opacity-25{--bs-link-opacity:0.25}.link-opacity-25-hover:hover{--bs-link-opacity:0.25}.link-opacity-50{--bs-link-opacity:0.5}.link-opacity-50-hover:hover{--bs-link-opacity:0.5}.link-opacity-75{--bs-link-opacity:0.75}.link-opacity-75-hover:hover{--bs-link-opacity:0.75}.link-opacity-100{--bs-link-opacity:1}.link-opacity-100-hover:hover{--bs-link-opacity:1}.link-offset-1{text-underline-offset:0.125em!important}.link-offset-1-hover:hover{text-underline-offset:0.125em!important}.link-offset-2{text-underline-offset:0.25em!important}.link-offset-2-hover:hover{text-underline-offset:0.25em!important}.link-offset-3{text-underline-offset:0.375em!important}.link-offset-3-hover:hover{text-underline-offset:0.375em!important}.link-underline-primary{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-primary-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-primary-rgb),var(--bs-link-underline-opacity))!important}.link-underline-secondary{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-secondary-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-secondary-rgb),var(--bs-link-underline-opacity))!important}.link-underline-success{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-success-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-success-rgb),var(--bs-link-underline-opacity))!important}.link-underline-info{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-info-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-info-rgb),var(--bs-link-underline-opacity))!important}.link-underline-warning{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-warning-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-warning-rgb),var(--bs-link-underline-opacity))!important}.link-underline-danger{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-danger-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-danger-rgb),var(--bs-link-underline-opacity))!important}.link-underline-light{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-light-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-light-rgb),var(--bs-link-underline-opacity))!important}.link-underline-dark{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-dark-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-dark-rgb),var(--bs-link-underline-opacity))!important}.link-underline{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-link-color-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:rgba(var(--bs-link-color-rgb),var(--bs-link-underline-opacity,1))!important}.link-underline-opacity-0{--bs-link-underline-opacity:0}.link-underline-opacity-0-hover:hover{--bs-link-underline-opacity:0}.link-underline-opacity-10{--bs-link-underline-opacity:0.1}.link-underline-opacity-10-hover:hover{--bs-link-underline-opacity:0.1}.link-underline-opacity-25{--bs-link-underline-opacity:0.25}.link-underline-opacity-25-hover:hover{--bs-link-underline-opacity:0.25}.link-underline-opacity-50{--bs-link-underline-opacity:0.5}.link-underline-opacity-50-hover:hover{--bs-link-underline-opacity:0.5}.link-underline-opacity-75{--bs-link-underline-opacity:0.75}.link-underline-opacity-75-hover:hover{--bs-link-underline-opacity:0.75}.link-underline-opacity-100{--bs-link-underline-opacity:1}.link-underline-opacity-100-hover:hover{--bs-link-underline-opacity:1}.bg-primary{--bs-bg-opacity:1;background-color:rgba(var(--bs-primary-rgb),var(--bs-bg-opacity))!important}.bg-secondary{--bs-bg-opacity:1;background-color:rgba(var(--bs-secondary-rgb),var(--bs-bg-opacity))!important}.bg-success{--bs-bg-opacity:1;background-color:rgba(var(--bs-success-rgb),var(--bs-bg-opacity))!important}.bg-info{--bs-bg-opacity:1;background-color:rgba(var(--bs-info-rgb),var(--bs-bg-opacity))!important}.bg-warning{--bs-bg-opacity:1;background-color:rgba(var(--bs-warning-rgb),var(--bs-bg-opacity))!important}.bg-danger{--bs-bg-opacity:1;background-color:rgba(var(--bs-danger-rgb),var(--bs-bg-opacity))!important}.bg-light{--bs-bg-opacity:1;background-color:rgba(var(--bs-light-rgb),var(--bs-bg-opacity))!important}.bg-dark{--bs-bg-opacity:1;background-color:rgba(var(--bs-dark-rgb),var(--bs-bg-opacity))!important}.bg-black{--bs-bg-opacity:1;background-color:rgba(var(--bs-black-rgb),var(--bs-bg-opacity))!important}.bg-white{--bs-bg-opacity:1;background-color:rgba(var(--bs-white-rgb),var(--bs-bg-opacity))!important}.bg-body{--bs-bg-opacity:1;background-color:rgba(var(--bs-body-bg-rgb),var(--bs-bg-opacity))!important}.bg-transparent{--bs-bg-opacity:1;background-color:transparent!important}.bg-body-secondary{--bs-bg-opacity:1;background-color:rgba(var(--bs-secondary-bg-rgb),var(--bs-bg-opacity))!important}.bg-body-tertiary{--bs-bg-opacity:1;background-color:rgba(var(--bs-tertiary-bg-rgb),var(--bs-bg-opacity))!important}.bg-opacity-10{--bs-bg-opacity:0.1}.bg-opacity-25{--bs-bg-opacity:0.25}.bg-opacity-50{--bs-bg-opacity:0.5}.bg-opacity-75{--bs-bg-opacity:0.75}.bg-opacity-100{--bs-bg-opacity:1}.bg-primary-subtle{background-color:var(--bs-primary-bg-subtle)!important}.bg-secondary-subtle{background-color:var(--bs-secondary-bg-subtle)!important}.bg-success-subtle{background-color:var(--bs-success-bg-subtle)!important}.bg-info-subtle{background-color:var(--bs-info-bg-subtle)!important}.bg-warning-subtle{background-color:var(--bs-warning-bg-subtle)!important}.bg-danger-subtle{background-color:var(--bs-danger-bg-subtle)!important}.bg-light-subtle{background-color:var(--bs-light-bg-subtle)!important}.bg-dark-subtle{background-color:var(--bs-dark-bg-subtle)!important}.bg-gradient{background-image:var(--bs-gradient)!important}.user-select-all{-webkit-user-select:all!important;-moz-user-select:all!important;user-select:all!important}.user-select-auto{-webkit-user-select:auto!important;-moz-user-select:auto!important;user-select:auto!important}.user-select-none{-webkit-user-select:none!important;-moz-user-select:none!important;user-select:none!important}.pe-none{pointer-events:none!important}.pe-auto{pointer-events:auto!important}.rounded{border-radius:var(--bs-border-radius)!important}.rounded-0{border-radius:0!important}.rounded-1{border-radius:var(--bs-border-radius-sm)!important}.rounded-2{border-radius:var(--bs-border-radius)!important}.rounded-3{border-radius:var(--bs-border-radius-lg)!important}.rounded-4{border-radius:var(--bs-border-radius-xl)!important}.rounded-5{border-radius:var(--bs-border-radius-xxl)!important}.rounded-circle{border-radius:50%!important}.rounded-pill{border-radius:var(--bs-border-radius-pill)!important}.rounded-top{border-top-left-radius:var(--bs-border-radius)!important;border-top-right-radius:var(--bs-border-radius)!important}.rounded-top-0{border-top-left-radius:0!important;border-top-right-radius:0!important}.rounded-top-1{border-top-left-radius:var(--bs-border-radius-sm)!important;border-top-right-radius:var(--bs-border-radius-sm)!important}.rounded-top-2{border-top-left-radius:var(--bs-border-radius)!important;border-top-right-radius:var(--bs-border-radius)!important}.rounded-top-3{border-top-left-radius:var(--bs-border-radius-lg)!important;border-top-right-radius:var(--bs-border-radius-lg)!important}.rounded-top-4{border-top-left-radius:var(--bs-border-radius-xl)!important;border-top-right-radius:var(--bs-border-radius-xl)!important}.rounded-top-5{border-top-left-radius:var(--bs-border-radius-xxl)!important;border-top-right-radius:var(--bs-border-radius-xxl)!important}.rounded-top-circle{border-top-left-radius:50%!important;border-top-right-radius:50%!important}.rounded-top-pill{border-top-left-radius:var(--bs-border-radius-pill)!important;border-top-right-radius:var(--bs-border-radius-pill)!important}.rounded-end{border-top-right-radius:var(--bs-border-radius)!important;border-bottom-right-radius:var(--bs-border-radius)!important}.rounded-end-0{border-top-right-radius:0!important;border-bottom-right-radius:0!important}.rounded-end-1{border-top-right-radius:var(--bs-border-radius-sm)!important;border-bottom-right-radius:var(--bs-border-radius-sm)!important}.rounded-end-2{border-top-right-radius:var(--bs-border-radius)!important;border-bottom-right-radius:var(--bs-border-radius)!important}.rounded-end-3{border-top-right-radius:var(--bs-border-radius-lg)!important;border-bottom-right-radius:var(--bs-border-radius-lg)!important}.rounded-end-4{border-top-right-radius:var(--bs-border-radius-xl)!important;border-bottom-right-radius:var(--bs-border-radius-xl)!important}.rounded-end-5{border-top-right-radius:var(--bs-border-radius-xxl)!important;border-bottom-right-radius:var(--bs-border-radius-xxl)!important}.rounded-end-circle{border-top-right-radius:50%!important;border-bottom-right-radius:50%!important}.rounded-end-pill{border-top-right-radius:var(--bs-border-radius-pill)!important;border-bottom-right-radius:var(--bs-border-radius-pill)!important}.rounded-bottom{border-bottom-right-radius:var(--bs-border-radius)!important;border-bottom-left-radius:var(--bs-border-radius)!important}.rounded-bottom-0{border-bottom-right-radius:0!important;border-bottom-left-radius:0!important}.rounded-bottom-1{border-bottom-right-radius:var(--bs-border-radius-sm)!important;border-bottom-left-radius:var(--bs-border-radius-sm)!important}.rounded-bottom-2{border-bottom-right-radius:var(--bs-border-radius)!important;border-bottom-left-radius:var(--bs-border-radius)!important}.rounded-bottom-3{border-bottom-right-radius:var(--bs-border-radius-lg)!important;border-bottom-left-radius:var(--bs-border-radius-lg)!important}.rounded-bottom-4{border-bottom-right-radius:var(--bs-border-radius-xl)!important;border-bottom-left-radius:var(--bs-border-radius-xl)!important}.rounded-bottom-5{border-bottom-right-radius:var(--bs-border-radius-xxl)!important;border-bottom-left-radius:var(--bs-border-radius-xxl)!important}.rounded-bottom-circle{border-bottom-right-radius:50%!important;border-bottom-left-radius:50%!important}.rounded-bottom-pill{border-bottom-right-radius:var(--bs-border-radius-pill)!important;border-bottom-left-radius:var(--bs-border-radius-pill)!important}.rounded-start{border-bottom-left-radius:var(--bs-border-radius)!important;border-top-left-radius:var(--bs-border-radius)!important}.rounded-start-0{border-bottom-left-radius:0!important;border-top-left-radius:0!important}.rounded-start-1{border-bottom-left-radius:var(--bs-border-radius-sm)!important;border-top-left-radius:var(--bs-border-radius-sm)!important}.rounded-start-2{border-bottom-left-radius:var(--bs-border-radius)!important;border-top-left-radius:var(--bs-border-radius)!important}.rounded-start-3{border-bottom-left-radius:var(--bs-border-radius-lg)!important;border-top-left-radius:var(--bs-border-radius-lg)!important}.rounded-start-4{border-bottom-left-radius:var(--bs-border-radius-xl)!important;border-top-left-radius:var(--bs-border-radius-xl)!important}.rounded-start-5{border-bottom-left-radius:var(--bs-border-radius-xxl)!important;border-top-left-radius:var(--bs-border-radius-xxl)!important}.rounded-start-circle{border-bottom-left-radius:50%!important;border-top-left-radius:50%!important}.rounded-start-pill{border-bottom-left-radius:var(--bs-border-radius-pill)!important;border-top-left-radius:var(--bs-border-radius-pill)!important}.visible{visibility:visible!important}.invisible{visibility:hidden!important}.z-n1{z-index:-1!important}.z-0{z-index:0!important}.z-1{z-index:1!important}.z-2{z-index:2!important}.z-3{z-index:3!important}@media (min-width:576px){.float-sm-start{float:left!important}.float-sm-end{float:right!important}.float-sm-none{float:none!important}.object-fit-sm-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-sm-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-sm-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-sm-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-sm-none{-o-object-fit:none!important;object-fit:none!important}.d-sm-inline{display:inline!important}.d-sm-inline-block{display:inline-block!important}.d-sm-block{display:block!important}.d-sm-grid{display:grid!important}.d-sm-inline-grid{display:inline-grid!important}.d-sm-table{display:table!important}.d-sm-table-row{display:table-row!important}.d-sm-table-cell{display:table-cell!important}.d-sm-flex{display:flex!important}.d-sm-inline-flex{display:inline-flex!important}.d-sm-none{display:none!important}.flex-sm-fill{flex:1 1 auto!important}.flex-sm-row{flex-direction:row!important}.flex-sm-column{flex-direction:column!important}.flex-sm-row-reverse{flex-direction:row-reverse!important}.flex-sm-column-reverse{flex-direction:column-reverse!important}.flex-sm-grow-0{flex-grow:0!important}.flex-sm-grow-1{flex-grow:1!important}.flex-sm-shrink-0{flex-shrink:0!important}.flex-sm-shrink-1{flex-shrink:1!important}.flex-sm-wrap{flex-wrap:wrap!important}.flex-sm-nowrap{flex-wrap:nowrap!important}.flex-sm-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-sm-start{justify-content:flex-start!important}.justify-content-sm-end{justify-content:flex-end!important}.justify-content-sm-center{justify-content:center!important}.justify-content-sm-between{justify-content:space-between!important}.justify-content-sm-around{justify-content:space-around!important}.justify-content-sm-evenly{justify-content:space-evenly!important}.align-items-sm-start{align-items:flex-start!important}.align-items-sm-end{align-items:flex-end!important}.align-items-sm-center{align-items:center!important}.align-items-sm-baseline{align-items:baseline!important}.align-items-sm-stretch{align-items:stretch!important}.align-content-sm-start{align-content:flex-start!important}.align-content-sm-end{align-content:flex-end!important}.align-content-sm-center{align-content:center!important}.align-content-sm-between{align-content:space-between!important}.align-content-sm-around{align-content:space-around!important}.align-content-sm-stretch{align-content:stretch!important}.align-self-sm-auto{align-self:auto!important}.align-self-sm-start{align-self:flex-start!important}.align-self-sm-end{align-self:flex-end!important}.align-self-sm-center{align-self:center!important}.align-self-sm-baseline{align-self:baseline!important}.align-self-sm-stretch{align-self:stretch!important}.order-sm-first{order:-1!important}.order-sm-0{order:0!important}.order-sm-1{order:1!important}.order-sm-2{order:2!important}.order-sm-3{order:3!important}.order-sm-4{order:4!important}.order-sm-5{order:5!important}.order-sm-last{order:6!important}.m-sm-0{margin:0!important}.m-sm-1{margin:.25rem!important}.m-sm-2{margin:.5rem!important}.m-sm-3{margin:1rem!important}.m-sm-4{margin:1.5rem!important}.m-sm-5{margin:3rem!important}.m-sm-auto{margin:auto!important}.mx-sm-0{margin-right:0!important;margin-left:0!important}.mx-sm-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-sm-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-sm-3{margin-right:1rem!important;margin-left:1rem!important}.mx-sm-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-sm-5{margin-right:3rem!important;margin-left:3rem!important}.mx-sm-auto{margin-right:auto!important;margin-left:auto!important}.my-sm-0{margin-top:0!important;margin-bottom:0!important}.my-sm-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-sm-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-sm-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-sm-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-sm-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-sm-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-sm-0{margin-top:0!important}.mt-sm-1{margin-top:.25rem!important}.mt-sm-2{margin-top:.5rem!important}.mt-sm-3{margin-top:1rem!important}.mt-sm-4{margin-top:1.5rem!important}.mt-sm-5{margin-top:3rem!important}.mt-sm-auto{margin-top:auto!important}.me-sm-0{margin-right:0!important}.me-sm-1{margin-right:.25rem!important}.me-sm-2{margin-right:.5rem!important}.me-sm-3{margin-right:1rem!important}.me-sm-4{margin-right:1.5rem!important}.me-sm-5{margin-right:3rem!important}.me-sm-auto{margin-right:auto!important}.mb-sm-0{margin-bottom:0!important}.mb-sm-1{margin-bottom:.25rem!important}.mb-sm-2{margin-bottom:.5rem!important}.mb-sm-3{margin-bottom:1rem!important}.mb-sm-4{margin-bottom:1.5rem!important}.mb-sm-5{margin-bottom:3rem!important}.mb-sm-auto{margin-bottom:auto!important}.ms-sm-0{margin-left:0!important}.ms-sm-1{margin-left:.25rem!important}.ms-sm-2{margin-left:.5rem!important}.ms-sm-3{margin-left:1rem!important}.ms-sm-4{margin-left:1.5rem!important}.ms-sm-5{margin-left:3rem!important}.ms-sm-auto{margin-left:auto!important}.p-sm-0{padding:0!important}.p-sm-1{padding:.25rem!important}.p-sm-2{padding:.5rem!important}.p-sm-3{padding:1rem!important}.p-sm-4{padding:1.5rem!important}.p-sm-5{padding:3rem!important}.px-sm-0{padding-right:0!important;padding-left:0!important}.px-sm-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-sm-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-sm-3{padding-right:1rem!important;padding-left:1rem!important}.px-sm-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-sm-5{padding-right:3rem!important;padding-left:3rem!important}.py-sm-0{padding-top:0!important;padding-bottom:0!important}.py-sm-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-sm-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-sm-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-sm-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-sm-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-sm-0{padding-top:0!important}.pt-sm-1{padding-top:.25rem!important}.pt-sm-2{padding-top:.5rem!important}.pt-sm-3{padding-top:1rem!important}.pt-sm-4{padding-top:1.5rem!important}.pt-sm-5{padding-top:3rem!important}.pe-sm-0{padding-right:0!important}.pe-sm-1{padding-right:.25rem!important}.pe-sm-2{padding-right:.5rem!important}.pe-sm-3{padding-right:1rem!important}.pe-sm-4{padding-right:1.5rem!important}.pe-sm-5{padding-right:3rem!important}.pb-sm-0{padding-bottom:0!important}.pb-sm-1{padding-bottom:.25rem!important}.pb-sm-2{padding-bottom:.5rem!important}.pb-sm-3{padding-bottom:1rem!important}.pb-sm-4{padding-bottom:1.5rem!important}.pb-sm-5{padding-bottom:3rem!important}.ps-sm-0{padding-left:0!important}.ps-sm-1{padding-left:.25rem!important}.ps-sm-2{padding-left:.5rem!important}.ps-sm-3{padding-left:1rem!important}.ps-sm-4{padding-left:1.5rem!important}.ps-sm-5{padding-left:3rem!important}.gap-sm-0{gap:0!important}.gap-sm-1{gap:.25rem!important}.gap-sm-2{gap:.5rem!important}.gap-sm-3{gap:1rem!important}.gap-sm-4{gap:1.5rem!important}.gap-sm-5{gap:3rem!important}.row-gap-sm-0{row-gap:0!important}.row-gap-sm-1{row-gap:.25rem!important}.row-gap-sm-2{row-gap:.5rem!important}.row-gap-sm-3{row-gap:1rem!important}.row-gap-sm-4{row-gap:1.5rem!important}.row-gap-sm-5{row-gap:3rem!important}.column-gap-sm-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-sm-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-sm-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-sm-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-sm-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-sm-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-sm-start{text-align:left!important}.text-sm-end{text-align:right!important}.text-sm-center{text-align:center!important}}@media (min-width:768px){.float-md-start{float:left!important}.float-md-end{float:right!important}.float-md-none{float:none!important}.object-fit-md-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-md-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-md-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-md-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-md-none{-o-object-fit:none!important;object-fit:none!important}.d-md-inline{display:inline!important}.d-md-inline-block{display:inline-block!important}.d-md-block{display:block!important}.d-md-grid{display:grid!important}.d-md-inline-grid{display:inline-grid!important}.d-md-table{display:table!important}.d-md-table-row{display:table-row!important}.d-md-table-cell{display:table-cell!important}.d-md-flex{display:flex!important}.d-md-inline-flex{display:inline-flex!important}.d-md-none{display:none!important}.flex-md-fill{flex:1 1 auto!important}.flex-md-row{flex-direction:row!important}.flex-md-column{flex-direction:column!important}.flex-md-row-reverse{flex-direction:row-reverse!important}.flex-md-column-reverse{flex-direction:column-reverse!important}.flex-md-grow-0{flex-grow:0!important}.flex-md-grow-1{flex-grow:1!important}.flex-md-shrink-0{flex-shrink:0!important}.flex-md-shrink-1{flex-shrink:1!important}.flex-md-wrap{flex-wrap:wrap!important}.flex-md-nowrap{flex-wrap:nowrap!important}.flex-md-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-md-start{justify-content:flex-start!important}.justify-content-md-end{justify-content:flex-end!important}.justify-content-md-center{justify-content:center!important}.justify-content-md-between{justify-content:space-between!important}.justify-content-md-around{justify-content:space-around!important}.justify-content-md-evenly{justify-content:space-evenly!important}.align-items-md-start{align-items:flex-start!important}.align-items-md-end{align-items:flex-end!important}.align-items-md-center{align-items:center!important}.align-items-md-baseline{align-items:baseline!important}.align-items-md-stretch{align-items:stretch!important}.align-content-md-start{align-content:flex-start!important}.align-content-md-end{align-content:flex-end!important}.align-content-md-center{align-content:center!important}.align-content-md-between{align-content:space-between!important}.align-content-md-around{align-content:space-around!important}.align-content-md-stretch{align-content:stretch!important}.align-self-md-auto{align-self:auto!important}.align-self-md-start{align-self:flex-start!important}.align-self-md-end{align-self:flex-end!important}.align-self-md-center{align-self:center!important}.align-self-md-baseline{align-self:baseline!important}.align-self-md-stretch{align-self:stretch!important}.order-md-first{order:-1!important}.order-md-0{order:0!important}.order-md-1{order:1!important}.order-md-2{order:2!important}.order-md-3{order:3!important}.order-md-4{order:4!important}.order-md-5{order:5!important}.order-md-last{order:6!important}.m-md-0{margin:0!important}.m-md-1{margin:.25rem!important}.m-md-2{margin:.5rem!important}.m-md-3{margin:1rem!important}.m-md-4{margin:1.5rem!important}.m-md-5{margin:3rem!important}.m-md-auto{margin:auto!important}.mx-md-0{margin-right:0!important;margin-left:0!important}.mx-md-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-md-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-md-3{margin-right:1rem!important;margin-left:1rem!important}.mx-md-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-md-5{margin-right:3rem!important;margin-left:3rem!important}.mx-md-auto{margin-right:auto!important;margin-left:auto!important}.my-md-0{margin-top:0!important;margin-bottom:0!important}.my-md-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-md-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-md-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-md-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-md-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-md-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-md-0{margin-top:0!important}.mt-md-1{margin-top:.25rem!important}.mt-md-2{margin-top:.5rem!important}.mt-md-3{margin-top:1rem!important}.mt-md-4{margin-top:1.5rem!important}.mt-md-5{margin-top:3rem!important}.mt-md-auto{margin-top:auto!important}.me-md-0{margin-right:0!important}.me-md-1{margin-right:.25rem!important}.me-md-2{margin-right:.5rem!important}.me-md-3{margin-right:1rem!important}.me-md-4{margin-right:1.5rem!important}.me-md-5{margin-right:3rem!important}.me-md-auto{margin-right:auto!important}.mb-md-0{margin-bottom:0!important}.mb-md-1{margin-bottom:.25rem!important}.mb-md-2{margin-bottom:.5rem!important}.mb-md-3{margin-bottom:1rem!important}.mb-md-4{margin-bottom:1.5rem!important}.mb-md-5{margin-bottom:3rem!important}.mb-md-auto{margin-bottom:auto!important}.ms-md-0{margin-left:0!important}.ms-md-1{margin-left:.25rem!important}.ms-md-2{margin-left:.5rem!important}.ms-md-3{margin-left:1rem!important}.ms-md-4{margin-left:1.5rem!important}.ms-md-5{margin-left:3rem!important}.ms-md-auto{margin-left:auto!important}.p-md-0{padding:0!important}.p-md-1{padding:.25rem!important}.p-md-2{padding:.5rem!important}.p-md-3{padding:1rem!important}.p-md-4{padding:1.5rem!important}.p-md-5{padding:3rem!important}.px-md-0{padding-right:0!important;padding-left:0!important}.px-md-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-md-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-md-3{padding-right:1rem!important;padding-left:1rem!important}.px-md-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-md-5{padding-right:3rem!important;padding-left:3rem!important}.py-md-0{padding-top:0!important;padding-bottom:0!important}.py-md-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-md-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-md-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-md-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-md-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-md-0{padding-top:0!important}.pt-md-1{padding-top:.25rem!important}.pt-md-2{padding-top:.5rem!important}.pt-md-3{padding-top:1rem!important}.pt-md-4{padding-top:1.5rem!important}.pt-md-5{padding-top:3rem!important}.pe-md-0{padding-right:0!important}.pe-md-1{padding-right:.25rem!important}.pe-md-2{padding-right:.5rem!important}.pe-md-3{padding-right:1rem!important}.pe-md-4{padding-right:1.5rem!important}.pe-md-5{padding-right:3rem!important}.pb-md-0{padding-bottom:0!important}.pb-md-1{padding-bottom:.25rem!important}.pb-md-2{padding-bottom:.5rem!important}.pb-md-3{padding-bottom:1rem!important}.pb-md-4{padding-bottom:1.5rem!important}.pb-md-5{padding-bottom:3rem!important}.ps-md-0{padding-left:0!important}.ps-md-1{padding-left:.25rem!important}.ps-md-2{padding-left:.5rem!important}.ps-md-3{padding-left:1rem!important}.ps-md-4{padding-left:1.5rem!important}.ps-md-5{padding-left:3rem!important}.gap-md-0{gap:0!important}.gap-md-1{gap:.25rem!important}.gap-md-2{gap:.5rem!important}.gap-md-3{gap:1rem!important}.gap-md-4{gap:1.5rem!important}.gap-md-5{gap:3rem!important}.row-gap-md-0{row-gap:0!important}.row-gap-md-1{row-gap:.25rem!important}.row-gap-md-2{row-gap:.5rem!important}.row-gap-md-3{row-gap:1rem!important}.row-gap-md-4{row-gap:1.5rem!important}.row-gap-md-5{row-gap:3rem!important}.column-gap-md-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-md-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-md-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-md-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-md-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-md-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-md-start{text-align:left!important}.text-md-end{text-align:right!important}.text-md-center{text-align:center!important}}@media (min-width:992px){.float-lg-start{float:left!important}.float-lg-end{float:right!important}.float-lg-none{float:none!important}.object-fit-lg-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-lg-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-lg-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-lg-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-lg-none{-o-object-fit:none!important;object-fit:none!important}.d-lg-inline{display:inline!important}.d-lg-inline-block{display:inline-block!important}.d-lg-block{display:block!important}.d-lg-grid{display:grid!important}.d-lg-inline-grid{display:inline-grid!important}.d-lg-table{display:table!important}.d-lg-table-row{display:table-row!important}.d-lg-table-cell{display:table-cell!important}.d-lg-flex{display:flex!important}.d-lg-inline-flex{display:inline-flex!important}.d-lg-none{display:none!important}.flex-lg-fill{flex:1 1 auto!important}.flex-lg-row{flex-direction:row!important}.flex-lg-column{flex-direction:column!important}.flex-lg-row-reverse{flex-direction:row-reverse!important}.flex-lg-column-reverse{flex-direction:column-reverse!important}.flex-lg-grow-0{flex-grow:0!important}.flex-lg-grow-1{flex-grow:1!important}.flex-lg-shrink-0{flex-shrink:0!important}.flex-lg-shrink-1{flex-shrink:1!important}.flex-lg-wrap{flex-wrap:wrap!important}.flex-lg-nowrap{flex-wrap:nowrap!important}.flex-lg-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-lg-start{justify-content:flex-start!important}.justify-content-lg-end{justify-content:flex-end!important}.justify-content-lg-center{justify-content:center!important}.justify-content-lg-between{justify-content:space-between!important}.justify-content-lg-around{justify-content:space-around!important}.justify-content-lg-evenly{justify-content:space-evenly!important}.align-items-lg-start{align-items:flex-start!important}.align-items-lg-end{align-items:flex-end!important}.align-items-lg-center{align-items:center!important}.align-items-lg-baseline{align-items:baseline!important}.align-items-lg-stretch{align-items:stretch!important}.align-content-lg-start{align-content:flex-start!important}.align-content-lg-end{align-content:flex-end!important}.align-content-lg-center{align-content:center!important}.align-content-lg-between{align-content:space-between!important}.align-content-lg-around{align-content:space-around!important}.align-content-lg-stretch{align-content:stretch!important}.align-self-lg-auto{align-self:auto!important}.align-self-lg-start{align-self:flex-start!important}.align-self-lg-end{align-self:flex-end!important}.align-self-lg-center{align-self:center!important}.align-self-lg-baseline{align-self:baseline!important}.align-self-lg-stretch{align-self:stretch!important}.order-lg-first{order:-1!important}.order-lg-0{order:0!important}.order-lg-1{order:1!important}.order-lg-2{order:2!important}.order-lg-3{order:3!important}.order-lg-4{order:4!important}.order-lg-5{order:5!important}.order-lg-last{order:6!important}.m-lg-0{margin:0!important}.m-lg-1{margin:.25rem!important}.m-lg-2{margin:.5rem!important}.m-lg-3{margin:1rem!important}.m-lg-4{margin:1.5rem!important}.m-lg-5{margin:3rem!important}.m-lg-auto{margin:auto!important}.mx-lg-0{margin-right:0!important;margin-left:0!important}.mx-lg-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-lg-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-lg-3{margin-right:1rem!important;margin-left:1rem!important}.mx-lg-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-lg-5{margin-right:3rem!important;margin-left:3rem!important}.mx-lg-auto{margin-right:auto!important;margin-left:auto!important}.my-lg-0{margin-top:0!important;margin-bottom:0!important}.my-lg-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-lg-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-lg-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-lg-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-lg-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-lg-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-lg-0{margin-top:0!important}.mt-lg-1{margin-top:.25rem!important}.mt-lg-2{margin-top:.5rem!important}.mt-lg-3{margin-top:1rem!important}.mt-lg-4{margin-top:1.5rem!important}.mt-lg-5{margin-top:3rem!important}.mt-lg-auto{margin-top:auto!important}.me-lg-0{margin-right:0!important}.me-lg-1{margin-right:.25rem!important}.me-lg-2{margin-right:.5rem!important}.me-lg-3{margin-right:1rem!important}.me-lg-4{margin-right:1.5rem!important}.me-lg-5{margin-right:3rem!important}.me-lg-auto{margin-right:auto!important}.mb-lg-0{margin-bottom:0!important}.mb-lg-1{margin-bottom:.25rem!important}.mb-lg-2{margin-bottom:.5rem!important}.mb-lg-3{margin-bottom:1rem!important}.mb-lg-4{margin-bottom:1.5rem!important}.mb-lg-5{margin-bottom:3rem!important}.mb-lg-auto{margin-bottom:auto!important}.ms-lg-0{margin-left:0!important}.ms-lg-1{margin-left:.25rem!important}.ms-lg-2{margin-left:.5rem!important}.ms-lg-3{margin-left:1rem!important}.ms-lg-4{margin-left:1.5rem!important}.ms-lg-5{margin-left:3rem!important}.ms-lg-auto{margin-left:auto!important}.p-lg-0{padding:0!important}.p-lg-1{padding:.25rem!important}.p-lg-2{padding:.5rem!important}.p-lg-3{padding:1rem!important}.p-lg-4{padding:1.5rem!important}.p-lg-5{padding:3rem!important}.px-lg-0{padding-right:0!important;padding-left:0!important}.px-lg-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-lg-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-lg-3{padding-right:1rem!important;padding-left:1rem!important}.px-lg-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-lg-5{padding-right:3rem!important;padding-left:3rem!important}.py-lg-0{padding-top:0!important;padding-bottom:0!important}.py-lg-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-lg-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-lg-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-lg-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-lg-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-lg-0{padding-top:0!important}.pt-lg-1{padding-top:.25rem!important}.pt-lg-2{padding-top:.5rem!important}.pt-lg-3{padding-top:1rem!important}.pt-lg-4{padding-top:1.5rem!important}.pt-lg-5{padding-top:3rem!important}.pe-lg-0{padding-right:0!important}.pe-lg-1{padding-right:.25rem!important}.pe-lg-2{padding-right:.5rem!important}.pe-lg-3{padding-right:1rem!important}.pe-lg-4{padding-right:1.5rem!important}.pe-lg-5{padding-right:3rem!important}.pb-lg-0{padding-bottom:0!important}.pb-lg-1{padding-bottom:.25rem!important}.pb-lg-2{padding-bottom:.5rem!important}.pb-lg-3{padding-bottom:1rem!important}.pb-lg-4{padding-bottom:1.5rem!important}.pb-lg-5{padding-bottom:3rem!important}.ps-lg-0{padding-left:0!important}.ps-lg-1{padding-left:.25rem!important}.ps-lg-2{padding-left:.5rem!important}.ps-lg-3{padding-left:1rem!important}.ps-lg-4{padding-left:1.5rem!important}.ps-lg-5{padding-left:3rem!important}.gap-lg-0{gap:0!important}.gap-lg-1{gap:.25rem!important}.gap-lg-2{gap:.5rem!important}.gap-lg-3{gap:1rem!important}.gap-lg-4{gap:1.5rem!important}.gap-lg-5{gap:3rem!important}.row-gap-lg-0{row-gap:0!important}.row-gap-lg-1{row-gap:.25rem!important}.row-gap-lg-2{row-gap:.5rem!important}.row-gap-lg-3{row-gap:1rem!important}.row-gap-lg-4{row-gap:1.5rem!important}.row-gap-lg-5{row-gap:3rem!important}.column-gap-lg-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-lg-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-lg-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-lg-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-lg-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-lg-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-lg-start{text-align:left!important}.text-lg-end{text-align:right!important}.text-lg-center{text-align:center!important}}@media (min-width:1200px){.float-xl-start{float:left!important}.float-xl-end{float:right!important}.float-xl-none{float:none!important}.object-fit-xl-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-xl-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-xl-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-xl-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-xl-none{-o-object-fit:none!important;object-fit:none!important}.d-xl-inline{display:inline!important}.d-xl-inline-block{display:inline-block!important}.d-xl-block{display:block!important}.d-xl-grid{display:grid!important}.d-xl-inline-grid{display:inline-grid!important}.d-xl-table{display:table!important}.d-xl-table-row{display:table-row!important}.d-xl-table-cell{display:table-cell!important}.d-xl-flex{display:flex!important}.d-xl-inline-flex{display:inline-flex!important}.d-xl-none{display:none!important}.flex-xl-fill{flex:1 1 auto!important}.flex-xl-row{flex-direction:row!important}.flex-xl-column{flex-direction:column!important}.flex-xl-row-reverse{flex-direction:row-reverse!important}.flex-xl-column-reverse{flex-direction:column-reverse!important}.flex-xl-grow-0{flex-grow:0!important}.flex-xl-grow-1{flex-grow:1!important}.flex-xl-shrink-0{flex-shrink:0!important}.flex-xl-shrink-1{flex-shrink:1!important}.flex-xl-wrap{flex-wrap:wrap!important}.flex-xl-nowrap{flex-wrap:nowrap!important}.flex-xl-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-xl-start{justify-content:flex-start!important}.justify-content-xl-end{justify-content:flex-end!important}.justify-content-xl-center{justify-content:center!important}.justify-content-xl-between{justify-content:space-between!important}.justify-content-xl-around{justify-content:space-around!important}.justify-content-xl-evenly{justify-content:space-evenly!important}.align-items-xl-start{align-items:flex-start!important}.align-items-xl-end{align-items:flex-end!important}.align-items-xl-center{align-items:center!important}.align-items-xl-baseline{align-items:baseline!important}.align-items-xl-stretch{align-items:stretch!important}.align-content-xl-start{align-content:flex-start!important}.align-content-xl-end{align-content:flex-end!important}.align-content-xl-center{align-content:center!important}.align-content-xl-between{align-content:space-between!important}.align-content-xl-around{align-content:space-around!important}.align-content-xl-stretch{align-content:stretch!important}.align-self-xl-auto{align-self:auto!important}.align-self-xl-start{align-self:flex-start!important}.align-self-xl-end{align-self:flex-end!important}.align-self-xl-center{align-self:center!important}.align-self-xl-baseline{align-self:baseline!important}.align-self-xl-stretch{align-self:stretch!important}.order-xl-first{order:-1!important}.order-xl-0{order:0!important}.order-xl-1{order:1!important}.order-xl-2{order:2!important}.order-xl-3{order:3!important}.order-xl-4{order:4!important}.order-xl-5{order:5!important}.order-xl-last{order:6!important}.m-xl-0{margin:0!important}.m-xl-1{margin:.25rem!important}.m-xl-2{margin:.5rem!important}.m-xl-3{margin:1rem!important}.m-xl-4{margin:1.5rem!important}.m-xl-5{margin:3rem!important}.m-xl-auto{margin:auto!important}.mx-xl-0{margin-right:0!important;margin-left:0!important}.mx-xl-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-xl-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-xl-3{margin-right:1rem!important;margin-left:1rem!important}.mx-xl-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-xl-5{margin-right:3rem!important;margin-left:3rem!important}.mx-xl-auto{margin-right:auto!important;margin-left:auto!important}.my-xl-0{margin-top:0!important;margin-bottom:0!important}.my-xl-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-xl-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-xl-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-xl-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-xl-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-xl-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-xl-0{margin-top:0!important}.mt-xl-1{margin-top:.25rem!important}.mt-xl-2{margin-top:.5rem!important}.mt-xl-3{margin-top:1rem!important}.mt-xl-4{margin-top:1.5rem!important}.mt-xl-5{margin-top:3rem!important}.mt-xl-auto{margin-top:auto!important}.me-xl-0{margin-right:0!important}.me-xl-1{margin-right:.25rem!important}.me-xl-2{margin-right:.5rem!important}.me-xl-3{margin-right:1rem!important}.me-xl-4{margin-right:1.5rem!important}.me-xl-5{margin-right:3rem!important}.me-xl-auto{margin-right:auto!important}.mb-xl-0{margin-bottom:0!important}.mb-xl-1{margin-bottom:.25rem!important}.mb-xl-2{margin-bottom:.5rem!important}.mb-xl-3{margin-bottom:1rem!important}.mb-xl-4{margin-bottom:1.5rem!important}.mb-xl-5{margin-bottom:3rem!important}.mb-xl-auto{margin-bottom:auto!important}.ms-xl-0{margin-left:0!important}.ms-xl-1{margin-left:.25rem!important}.ms-xl-2{margin-left:.5rem!important}.ms-xl-3{margin-left:1rem!important}.ms-xl-4{margin-left:1.5rem!important}.ms-xl-5{margin-left:3rem!important}.ms-xl-auto{margin-left:auto!important}.p-xl-0{padding:0!important}.p-xl-1{padding:.25rem!important}.p-xl-2{padding:.5rem!important}.p-xl-3{padding:1rem!important}.p-xl-4{padding:1.5rem!important}.p-xl-5{padding:3rem!important}.px-xl-0{padding-right:0!important;padding-left:0!important}.px-xl-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-xl-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-xl-3{padding-right:1rem!important;padding-left:1rem!important}.px-xl-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-xl-5{padding-right:3rem!important;padding-left:3rem!important}.py-xl-0{padding-top:0!important;padding-bottom:0!important}.py-xl-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-xl-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-xl-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-xl-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-xl-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-xl-0{padding-top:0!important}.pt-xl-1{padding-top:.25rem!important}.pt-xl-2{padding-top:.5rem!important}.pt-xl-3{padding-top:1rem!important}.pt-xl-4{padding-top:1.5rem!important}.pt-xl-5{padding-top:3rem!important}.pe-xl-0{padding-right:0!important}.pe-xl-1{padding-right:.25rem!important}.pe-xl-2{padding-right:.5rem!important}.pe-xl-3{padding-right:1rem!important}.pe-xl-4{padding-right:1.5rem!important}.pe-xl-5{padding-right:3rem!important}.pb-xl-0{padding-bottom:0!important}.pb-xl-1{padding-bottom:.25rem!important}.pb-xl-2{padding-bottom:.5rem!important}.pb-xl-3{padding-bottom:1rem!important}.pb-xl-4{padding-bottom:1.5rem!important}.pb-xl-5{padding-bottom:3rem!important}.ps-xl-0{padding-left:0!important}.ps-xl-1{padding-left:.25rem!important}.ps-xl-2{padding-left:.5rem!important}.ps-xl-3{padding-left:1rem!important}.ps-xl-4{padding-left:1.5rem!important}.ps-xl-5{padding-left:3rem!important}.gap-xl-0{gap:0!important}.gap-xl-1{gap:.25rem!important}.gap-xl-2{gap:.5rem!important}.gap-xl-3{gap:1rem!important}.gap-xl-4{gap:1.5rem!important}.gap-xl-5{gap:3rem!important}.row-gap-xl-0{row-gap:0!important}.row-gap-xl-1{row-gap:.25rem!important}.row-gap-xl-2{row-gap:.5rem!important}.row-gap-xl-3{row-gap:1rem!important}.row-gap-xl-4{row-gap:1.5rem!important}.row-gap-xl-5{row-gap:3rem!important}.column-gap-xl-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-xl-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-xl-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-xl-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-xl-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-xl-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-xl-start{text-align:left!important}.text-xl-end{text-align:right!important}.text-xl-center{text-align:center!important}}@media (min-width:1400px){.float-xxl-start{float:left!important}.float-xxl-end{float:right!important}.float-xxl-none{float:none!important}.object-fit-xxl-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-xxl-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-xxl-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-xxl-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-xxl-none{-o-object-fit:none!important;object-fit:none!important}.d-xxl-inline{display:inline!important}.d-xxl-inline-block{display:inline-block!important}.d-xxl-block{display:block!important}.d-xxl-grid{display:grid!important}.d-xxl-inline-grid{display:inline-grid!important}.d-xxl-table{display:table!important}.d-xxl-table-row{display:table-row!important}.d-xxl-table-cell{display:table-cell!important}.d-xxl-flex{display:flex!important}.d-xxl-inline-flex{display:inline-flex!important}.d-xxl-none{display:none!important}.flex-xxl-fill{flex:1 1 auto!important}.flex-xxl-row{flex-direction:row!important}.flex-xxl-column{flex-direction:column!important}.flex-xxl-row-reverse{flex-direction:row-reverse!important}.flex-xxl-column-reverse{flex-direction:column-reverse!important}.flex-xxl-grow-0{flex-grow:0!important}.flex-xxl-grow-1{flex-grow:1!important}.flex-xxl-shrink-0{flex-shrink:0!important}.flex-xxl-shrink-1{flex-shrink:1!important}.flex-xxl-wrap{flex-wrap:wrap!important}.flex-xxl-nowrap{flex-wrap:nowrap!important}.flex-xxl-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-xxl-start{justify-content:flex-start!important}.justify-content-xxl-end{justify-content:flex-end!important}.justify-content-xxl-center{justify-content:center!important}.justify-content-xxl-between{justify-content:space-between!important}.justify-content-xxl-around{justify-content:space-around!important}.justify-content-xxl-evenly{justify-content:space-evenly!important}.align-items-xxl-start{align-items:flex-start!important}.align-items-xxl-end{align-items:flex-end!important}.align-items-xxl-center{align-items:center!important}.align-items-xxl-baseline{align-items:baseline!important}.align-items-xxl-stretch{align-items:stretch!important}.align-content-xxl-start{align-content:flex-start!important}.align-content-xxl-end{align-content:flex-end!important}.align-content-xxl-center{align-content:center!important}.align-content-xxl-between{align-content:space-between!important}.align-content-xxl-around{align-content:space-around!important}.align-content-xxl-stretch{align-content:stretch!important}.align-self-xxl-auto{align-self:auto!important}.align-self-xxl-start{align-self:flex-start!important}.align-self-xxl-end{align-self:flex-end!important}.align-self-xxl-center{align-self:center!important}.align-self-xxl-baseline{align-self:baseline!important}.align-self-xxl-stretch{align-self:stretch!important}.order-xxl-first{order:-1!important}.order-xxl-0{order:0!important}.order-xxl-1{order:1!important}.order-xxl-2{order:2!important}.order-xxl-3{order:3!important}.order-xxl-4{order:4!important}.order-xxl-5{order:5!important}.order-xxl-last{order:6!important}.m-xxl-0{margin:0!important}.m-xxl-1{margin:.25rem!important}.m-xxl-2{margin:.5rem!important}.m-xxl-3{margin:1rem!important}.m-xxl-4{margin:1.5rem!important}.m-xxl-5{margin:3rem!important}.m-xxl-auto{margin:auto!important}.mx-xxl-0{margin-right:0!important;margin-left:0!important}.mx-xxl-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-xxl-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-xxl-3{margin-right:1rem!important;margin-left:1rem!important}.mx-xxl-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-xxl-5{margin-right:3rem!important;margin-left:3rem!important}.mx-xxl-auto{margin-right:auto!important;margin-left:auto!important}.my-xxl-0{margin-top:0!important;margin-bottom:0!important}.my-xxl-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-xxl-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-xxl-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-xxl-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-xxl-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-xxl-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-xxl-0{margin-top:0!important}.mt-xxl-1{margin-top:.25rem!important}.mt-xxl-2{margin-top:.5rem!important}.mt-xxl-3{margin-top:1rem!important}.mt-xxl-4{margin-top:1.5rem!important}.mt-xxl-5{margin-top:3rem!important}.mt-xxl-auto{margin-top:auto!important}.me-xxl-0{margin-right:0!important}.me-xxl-1{margin-right:.25rem!important}.me-xxl-2{margin-right:.5rem!important}.me-xxl-3{margin-right:1rem!important}.me-xxl-4{margin-right:1.5rem!important}.me-xxl-5{margin-right:3rem!important}.me-xxl-auto{margin-right:auto!important}.mb-xxl-0{margin-bottom:0!important}.mb-xxl-1{margin-bottom:.25rem!important}.mb-xxl-2{margin-bottom:.5rem!important}.mb-xxl-3{margin-bottom:1rem!important}.mb-xxl-4{margin-bottom:1.5rem!important}.mb-xxl-5{margin-bottom:3rem!important}.mb-xxl-auto{margin-bottom:auto!important}.ms-xxl-0{margin-left:0!important}.ms-xxl-1{margin-left:.25rem!important}.ms-xxl-2{margin-left:.5rem!important}.ms-xxl-3{margin-left:1rem!important}.ms-xxl-4{margin-left:1.5rem!important}.ms-xxl-5{margin-left:3rem!important}.ms-xxl-auto{margin-left:auto!important}.p-xxl-0{padding:0!important}.p-xxl-1{padding:.25rem!important}.p-xxl-2{padding:.5rem!important}.p-xxl-3{padding:1rem!important}.p-xxl-4{padding:1.5rem!important}.p-xxl-5{padding:3rem!important}.px-xxl-0{padding-right:0!important;padding-left:0!important}.px-xxl-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-xxl-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-xxl-3{padding-right:1rem!important;padding-left:1rem!important}.px-xxl-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-xxl-5{padding-right:3rem!important;padding-left:3rem!important}.py-xxl-0{padding-top:0!important;padding-bottom:0!important}.py-xxl-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-xxl-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-xxl-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-xxl-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-xxl-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-xxl-0{padding-top:0!important}.pt-xxl-1{padding-top:.25rem!important}.pt-xxl-2{padding-top:.5rem!important}.pt-xxl-3{padding-top:1rem!important}.pt-xxl-4{padding-top:1.5rem!important}.pt-xxl-5{padding-top:3rem!important}.pe-xxl-0{padding-right:0!important}.pe-xxl-1{padding-right:.25rem!important}.pe-xxl-2{padding-right:.5rem!important}.pe-xxl-3{padding-right:1rem!important}.pe-xxl-4{padding-right:1.5rem!important}.pe-xxl-5{padding-right:3rem!important}.pb-xxl-0{padding-bottom:0!important}.pb-xxl-1{padding-bottom:.25rem!important}.pb-xxl-2{padding-bottom:.5rem!important}.pb-xxl-3{padding-bottom:1rem!important}.pb-xxl-4{padding-bottom:1.5rem!important}.pb-xxl-5{padding-bottom:3rem!important}.ps-xxl-0{padding-left:0!important}.ps-xxl-1{padding-left:.25rem!important}.ps-xxl-2{padding-left:.5rem!important}.ps-xxl-3{padding-left:1rem!important}.ps-xxl-4{padding-left:1.5rem!important}.ps-xxl-5{padding-left:3rem!important}.gap-xxl-0{gap:0!important}.gap-xxl-1{gap:.25rem!important}.gap-xxl-2{gap:.5rem!important}.gap-xxl-3{gap:1rem!important}.gap-xxl-4{gap:1.5rem!important}.gap-xxl-5{gap:3rem!important}.row-gap-xxl-0{row-gap:0!important}.row-gap-xxl-1{row-gap:.25rem!important}.row-gap-xxl-2{row-gap:.5rem!important}.row-gap-xxl-3{row-gap:1rem!important}.row-gap-xxl-4{row-gap:1.5rem!important}.row-gap-xxl-5{row-gap:3rem!important}.column-gap-xxl-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-xxl-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-xxl-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-xxl-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-xxl-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-xxl-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-xxl-start{text-align:left!important}.text-xxl-end{text-align:right!important}.text-xxl-center{text-align:center!important}}@media (min-width:1200px){.fs-1{font-size:2.5rem!important}.fs-2{font-size:2rem!important}.fs-3{font-size:1.75rem!important}.fs-4{font-size:1.5rem!important}}@media print{.d-print-inline{display:inline!important}.d-print-inline-block{display:inline-block!important}.d-print-block{display:block!important}.d-print-grid{display:grid!important}.d-print-inline-grid{display:inline-grid!important}.d-print-table{display:table!important}.d-print-table-row{display:table-row!important}.d-print-table-cell{display:table-cell!important}.d-print-flex{display:flex!important}.d-print-inline-flex{display:inline-flex!important}.d-print-none{display:none!important}} diff --git a/src/main/webapp/static/css/Bold-BS4-Full-Page-Image-Header.css b/src/main/webapp/static/css/Bold-BS4-Full-Page-Image-Header.css new file mode 100644 index 0000000..808140b --- /dev/null +++ b/src/main/webapp/static/css/Bold-BS4-Full-Page-Image-Header.css @@ -0,0 +1,95 @@ +.masthead { + height: 100vh; + min-height: 500px; + background-image: url('https://source.unsplash.com/BtbjCFUvBXs/1920x1080'); + background-size: cover; + background-position: center; + background-repeat: no-repeat; +} + +.col { + display: flex; + flex-direction: row; + justify-content: center; +} + +.button-group { + display: flex; + flex-direction: column; + gap: 5px; + margin-top: 20px; +} + +.textarea-container { + width: 100%; + height: 100vh; + display: flex; + justify-content: center; + align-items: flex-start; + padding: 20px 0; + box-sizing: border-box; +} + +@media (max-width: 768px) { + .text-white { + width: 95%; + margin: 0 10px; + } +} + +textarea { + width: 30vw; + height: 50vh; + min-width: 300px; + min-height: 150px; + padding: 12px; + font-size: 16px; + line-height: 1.5; + border: 2px solid #007bff; + border-radius: 8px; + resize: none; + box-sizing: border-box; + font-family: sans-serif; +} + +textarea:focus { + outline: none; + border-color: #0056b3; + box-shadow: 0 0 0 3px rgba(0, 123, 255, 0.25); +} + +input[type="radio"] { + border: 2px solid #007BFF; + border-radius: 50%; + width: 16px; + height: 16px; +} + +input[type="radio"]:hover { + border-color: #0056b3; +} + +.row { + row-gap: 20px; +} + +.edit-mode { + display: none; +} + +.d-none { + display: none; +} + +.form-group { + margin-bottom: 20px; +} + +.question-text { + font-size: large; + font-weight: bold; + color: #0b5ed7; +} + + + diff --git a/src/main/webapp/static/css/Hero-Features-icons.css b/src/main/webapp/static/css/Hero-Features-icons.css new file mode 100644 index 0000000..b6e79a3 --- /dev/null +++ b/src/main/webapp/static/css/Hero-Features-icons.css @@ -0,0 +1,57 @@ +.bs-icon { + --bs-icon-size: .75rem; + display: flex; + flex-shrink: 0; + justify-content: center; + align-items: center; + font-size: var(--bs-icon-size); + width: calc(var(--bs-icon-size) * 2); + height: calc(var(--bs-icon-size) * 2); + color: var(--bs-primary); +} + +.bs-icon-xs { + --bs-icon-size: 1rem; + width: calc(var(--bs-icon-size) * 1.5); + height: calc(var(--bs-icon-size) * 1.5); +} + +.bs-icon-sm { + --bs-icon-size: 1rem; +} + +.bs-icon-md { + --bs-icon-size: 1.5rem; +} + +.bs-icon-lg { + --bs-icon-size: 2rem; +} + +.bs-icon-xl { + --bs-icon-size: 2.5rem; +} + +.bs-icon.bs-icon-primary { + color: var(--bs-white); + background: var(--bs-primary); +} + +.bs-icon.bs-icon-primary-light { + color: var(--bs-primary); + background: rgba(var(--bs-primary-rgb), .2); +} + +.bs-icon.bs-icon-semi-white { + color: var(--bs-primary); + background: rgba(255, 255, 255, .5); +} + +.bs-icon.bs-icon-rounded { + border-radius: .5rem; +} + +.bs-icon.bs-icon-circle { + border-radius: 50%; +} + diff --git a/src/main/webapp/static/css/style.css b/src/main/webapp/static/css/style.css new file mode 100644 index 0000000..f1a5926 --- /dev/null +++ b/src/main/webapp/static/css/style.css @@ -0,0 +1,112 @@ +/* Базовый сброс отступов и полей */ +* { + margin: 0; + padding: 0; + box-sizing: border-box; + text-align: center; +} + +body { + font-family: 'Georgia', serif; + background-color: #1a110a; + color: #e6d4b0; + line-height: 1.6; + padding: 20px; + margin: 0 auto; + background-image: linear-gradient(to bottom, #1a110a, #0d0805); +} + +h1 { + font-size: 2.5rem; + text-align: center; + color: #d4a56c; + text-shadow: 2px 2px 4px rgba(0, 0, 0, 0.6); + margin-bottom: 30px; + letter-spacing: 1px; +} + +p { + font-size: 1.1rem; + margin-bottom: 20px; + text-align: justify; +} + +b { + color: #f0c48c; + font-weight: bold; +} + +a { + text-decoration: none; + color: #d4b588; +} + +a:hover { + text-decoration: none; + color: #fff; + font-size: large; +} + +button { + display: block; + width: 220px; + padding: 12px 20px; + margin: 20px auto; + border: none; + border-radius: 6px; + font-family: 'Georgia', serif; + font-size: 1rem; + cursor: pointer; + transition: all 0.3s ease; + text-transform: uppercase; + letter-spacing: 1px; +} + +button:first-of-type { + background-color: #8b4513; + color: #fff; + box-shadow: 0 4px 6px rgba(139, 69, 19, 0.4); +} + +button:first-of-type:hover { + background-color: #a0522d; + transform: translateY(-2px); +} + +button:last-of-type { + background-color: #4a3a2a; + color: #d4b588; + box-shadow: 0 4px 6px rgba(74, 58, 42, 0.4); +} + +button:last-of-type:hover { + background-color: #5e4c3a; + transform: translateY(-2px); +} + +.quest-container { + margin: 30px; +} + +.quest-text { + background-color: rgba(34, 24, 16, 0.6); + padding: 20px; + border-radius: 8px; + border: 1px solid rgba(102, 71, 45, 0.5); + box-shadow: 0 0 15px rgba(102, 71, 45, 0.3); + color: #e6d4b0; +} + +@media (max-width: 600px) { + h1 { + font-size: 2rem; + } + p { + font-size: 1rem; + padding: 0 10px; + } + button { + width: 180px; + font-size: 0.9rem; + } +} diff --git a/src/test/java/com/javarush/vasileva/BaseIT.java b/src/test/java/com/javarush/vasileva/BaseIT.java new file mode 100644 index 0000000..8dd5c21 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/BaseIT.java @@ -0,0 +1,225 @@ +package com.javarush.vasileva; + +import com.javarush.vasileva.config.Config; +import com.javarush.vasileva.config.Winter; +import com.javarush.vasileva.entity.*; +import jakarta.servlet.ServletConfig; +import jakarta.servlet.ServletContext; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpServletResponse; +import jakarta.servlet.http.HttpSession; +import org.mockito.Mockito; +import org.slf4j.Logger; + +import java.util.List; + +import static org.mockito.Mockito.when; + +public class BaseIT { + protected final HttpServletRequest req; + protected final HttpServletResponse resp; + protected final HttpSession session; + protected final Config config; + protected final ServletConfig servletConfig; + protected final ServletContext servletContext; + protected final Logger LOGGER; + + protected User testAdmin; + protected User testUser; + protected User testGuest; + + protected Game testGame; + protected GameState testGameState; + + protected Quest testQuest; + protected Quest testQuestWithoutId; + + protected Question testQuestion1; + protected Question testQuestion2; + protected Question testQuestion3; + + protected Answer testAnswer1; + protected Answer testAnswer2; + + protected String testJson; + protected String testJsonWithoutId; + + protected UserStats testUserStats; + + public BaseIT() { + config = Winter.find(Config.class); + config.fillRepository(); + + servletConfig = Mockito.mock(ServletConfig.class); + servletContext = Mockito.mock(ServletContext.class); + when(servletConfig.getServletContext()).thenReturn(servletContext); + + req = Mockito.mock(HttpServletRequest.class); + resp = Mockito.mock(HttpServletResponse.class); + session = Mockito.mock(HttpSession.class); + when(req.getSession()).thenReturn(session); + + LOGGER = Mockito.mock(Logger.class); + + testAdmin = User.builder() + .id(1L) + .login("testAdmin") + .email("admin@test.com") + .password("testAdmin") + .role(Role.ADMIN) + .build(); + + testUser = User.builder() + .id(2L) + .login("testUser") + .email("user@test.com") + .password("testUser") + .role(Role.USER) + .build(); + + testGuest = User.builder() + .id(3L) + .login("testGuest") + .email("guest@test.com") + .password("testGuest") + .role(Role.GUEST) + .build(); + + testAnswer1 = Answer.builder() + .id(1L) + .questionId(1L) + .text("testAnswer1") + .nextQuestionLabel("+8") + .build(); + + testAnswer2 = Answer.builder() + .id(2L) + .questionId(1L) + .text("testAnswer2") + .nextQuestionLabel("-9") + .build(); + + testQuestion1 = Question.builder() + .generatedId(1L) + .questId(1L) + .label("1") + .text("testQuestion1") + .answers(List.of(testAnswer1, testAnswer2)) + .build(); + + testQuestion2 = Question.builder() + .generatedId(2L) + .label("+8") + .text("testQuestion2") + .build(); + + testQuestion3 = Question.builder() + .generatedId(3L) + .label("-9") + .text("testQuestion3") + .build(); + + testQuest = Quest.builder() + .id(1L) + .title("testQuest") + .description("testQuest") + .text("testQuest") + .startQuestionId(1L) + .questions(List.of(testQuestion1, testQuestion2, testQuestion3)) + .build(); + + testQuestWithoutId = Quest.builder() + .title("testQuest") + .description("testQuest") + .text("testQuest") + .startQuestionId(1L) + .questions(List.of(testQuestion1, testQuestion2, testQuestion3)) + .build(); + + testGameState = GameState.builder() + .currentQuest(testQuest) + .currentQuestion(testQuestion1) + .user(testUser) + .isCompleted(false) + .build(); + + testGame = Game.builder() + .id(1L) + .questId(testQuest.getId()) + .userId(testUser.getId()) + .currentQuestionId(testQuestion1.getGeneratedId()) + .gameState(testGameState) + .build(); + + testJson = """ + { + "id": 1, + "title": "testQuest", + "description": "testQuest", + "text": "testQuest", + "questions": [ + { + "label": "1", + "text": "testQuestion1", + "answers": [ + { + "nextQuestionLabel": "+8", + "text": "testAnswer1" + }, + { + "nextQuestionLabel": "-9", + "text": "testAnswer2" + } + ] + }, + { + "label": "+8", + "text": "testQuestion2" + }, + { + "label": "-9", + "text": "testQuestion2" + } + ] + }"""; + + testJsonWithoutId = """ + { + "title": "testQuest", + "description": "testQuest", + "text": "testQuest", + "questions": [ + { + "label": "1", + "text": "testQuestion1", + "answers": [ + { + "nextQuestionLabel": "+8", + "text": "testAnswer1" + }, + { + "nextQuestionLabel": "-9", + "text": "testAnswer2" + } + ] + }, + { + "label": "+8", + "text": "testQuestion2" + }, + { + "label": "-9", + "text": "testQuestion2" + } + ] + }"""; + + testUserStats = UserStats.builder() + .id(1L) + .userId(testUser.getId()) + .total(3) + .wins(2) + .losses(1) + .build(); + } +} diff --git a/src/test/java/com/javarush/vasileva/cmd/EditQuestIT.java b/src/test/java/com/javarush/vasileva/cmd/EditQuestIT.java new file mode 100644 index 0000000..87a8d4f --- /dev/null +++ b/src/test/java/com/javarush/vasileva/cmd/EditQuestIT.java @@ -0,0 +1,175 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.BaseIT; +import com.javarush.vasileva.entity.Quest; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.mapper.QuestMapper; +import com.javarush.vasileva.service.AuthService; +import com.javarush.vasileva.service.QuestService; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.mockito.ArgumentCaptor; + +import java.io.IOException; +import java.util.Collections; +import java.util.List; +import java.util.Optional; + +import static com.javarush.vasileva.util.Key.*; +import static com.javarush.vasileva.util.Link.HOME; +import static com.javarush.vasileva.util.Value.*; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +public class EditQuestIT extends BaseIT { + + private final QuestService questService = mock(QuestService.class); + private final AuthService authService = mock(AuthService.class); + private final QuestMapper questMapper = mock(QuestMapper.class); + + private final EditQuest editQuest = new EditQuest(questService, authService, config); + + @Test + @DisplayName("when GET request admin authorized no questId then set edit false and default JSON") + void whenGetNoQuestId_ThenSetEditFalseAndDefaultJson() { + doNothing().when(authService).checkAdminAuthorization(req, EDIT_QUEST_AUTH_ERROR); + when(req.getParameter(QUEST_ID)).thenReturn(null); + when(questService.getAll()).thenReturn(Collections.singletonList(testQuest)); + + String view = editQuest.doGet(req); + + assertEquals(editQuest.getView(), view); + verify(authService).checkAdminAuthorization(req, EDIT_QUEST_AUTH_ERROR); + verify(req).setAttribute(eq(QUESTS), anyList()); + verify(req).setAttribute(eq(EDIT), eq(false)); + verify(req).setAttribute(eq(QUEST_JSON), eq(JSON_SAMPLE)); + } + + @Test + @DisplayName("when POST request with null JSON parameter then set error in session") + void whenPostNullJsonParameterThenSetErrorInSession() { + when(req.getParameter(QUEST_JSON)).thenReturn(null); + + String view = editQuest.doPost(req); + + assertEquals(editQuest.getView(), view); + verify(session).setAttribute(eq(ERROR), eq(JSON_SAVE_ERROR)); + verify(session, never()).setAttribute(eq(QUEST_JSON), any()); + + verifyNoInteractions(questMapper); + verifyNoInteractions(questService); + } + + @Test + @DisplayName("when JSON serialization fails in doGet then return view") + void whenJsonSerializationFailsInDoGet_ThenReturnViewAndLogError() throws Exception { + doNothing().when(authService).checkAdminAuthorization(req, EDIT_QUEST_AUTH_ERROR); + + when(req.getParameter(QUEST_ID)).thenReturn(String.valueOf(testQuest.getId())); + when(questService.getValidatedQuest(String.valueOf(testQuest.getId()))) + .thenReturn(Optional.of(testQuest)); + + when(questMapper.toJsonString(testQuest)) + .thenThrow(new IOException("Serialization failed")); + + String resultView = editQuest.doGet(req); + + assertEquals(editQuest.getView(), resultView); + verify(req).setAttribute(eq(EDIT), eq(true)); + assertNull(req.getAttribute(QUEST_JSON)); + assertThrows(IOException.class, () -> questMapper.toJsonString(testQuest)); + } + + @Test + @DisplayName("when GET request but quest not found then throw AppException") + void whenGetQuestNotFound_ThenThrowAppException() { + String questIdStr = "999"; + doNothing().when(authService).checkAdminAuthorization(req, EDIT_QUEST_AUTH_ERROR); + when(req.getParameter(QUEST_ID)).thenReturn(questIdStr); + when(questService.getValidatedQuest(questIdStr)).thenReturn(Optional.empty()); + + when(questService.getAll()).thenReturn(List.of()); + + AppException exception = assertThrows(AppException.class, () -> editQuest.doGet(req)); + + assertEquals(QUEST_NOT_FOUND + questIdStr, exception.getMessage()); + verify(questService).getValidatedQuest(questIdStr); + } + + @Test + @DisplayName("when GET request but serialization fails then set error in session") + void whenGetSerializationFails_ThenSetErrorInSession() throws Exception { + doNothing().when(authService).checkAdminAuthorization(req, EDIT_QUEST_AUTH_ERROR); + when(req.getParameter(QUEST_ID)).thenReturn(String.valueOf(testQuest.getId())); + when(questService.getValidatedQuest(String.valueOf(testQuest.getId()))) + .thenReturn(Optional.of(testQuest)); + when(questMapper.toJsonString(testQuest)).thenThrow(new IOException("Serialization failed")); + when(questService.getAll()).thenReturn(Collections.singletonList(testQuest)); + + String view = editQuest.doGet(req); + + assertEquals(editQuest.getView(), view); + } + + @Test + @DisplayName("when POST request and JSON valid then save quest and redirect to HOME") + void whenPostValidJson_ThenSaveQuestAndRedirect() throws Exception { + when(req.getParameter(QUEST_JSON)).thenReturn(testJsonWithoutId); + when(questMapper.fromJsonString(testJson)).thenReturn(testQuestWithoutId); + + String redirect = editQuest.doPost(req); + + assertEquals(HOME, redirect); + verify(questService).create(any(Quest.class)); + + ArgumentCaptor questCaptor = ArgumentCaptor.forClass(Quest.class); + verify(questService).create(questCaptor.capture()); + } + + @Test + @DisplayName("when POST request with existing quest ID then update quest") + void whenPostWithExistingId_ThenUpdateQuest() throws Exception { + when(req.getParameter(QUEST_JSON)).thenReturn(testJson); + when(questMapper.fromJsonString(testJson)).thenReturn(testQuest); + + String redirect = editQuest.doPost(req); + + assertEquals(HOME, redirect); + verify(questService).update(any(Quest.class)); + + ArgumentCaptor questCaptor = ArgumentCaptor.forClass(Quest.class); + verify(questService).update(questCaptor.capture()); + } + + @Test + @DisplayName("when POST request but JSON invalid then set error in session") + void whenPostInvalidJson_ThenSetErrorInSession() throws Exception { + String questJson = "invalid json"; + when(req.getParameter(QUEST_JSON)).thenReturn(questJson); + when(questMapper.fromJsonString(questJson)) + .thenThrow(new IOException("Parse failed")); + + String view = editQuest.doPost(req); + + assertEquals(editQuest.getView(), view); + + verify(session).setAttribute(eq(QUEST_JSON), eq(questJson)); + verify(session).setAttribute(eq(ERROR), eq(JSON_SAVE_ERROR)); + } + + + @Test + @DisplayName("When GET request but admin authorization fails then throw AppException") + void whenGetAdminAuthorizationFails_ThenThrowAppException() { + doThrow(new AppException(EDIT_QUEST_AUTH_ERROR)) + .when(authService) + .checkAdminAuthorization(req, EDIT_QUEST_AUTH_ERROR); + + AppException exception = assertThrows(AppException.class, () -> editQuest.doGet(req)); + + assertEquals(EDIT_QUEST_AUTH_ERROR, exception.getMessage()); + verify(authService).checkAdminAuthorization(req, EDIT_QUEST_AUTH_ERROR); + verifyNoInteractions(questService); + verifyNoInteractions(req); + } +} diff --git a/src/test/java/com/javarush/vasileva/cmd/EditUserIT.java b/src/test/java/com/javarush/vasileva/cmd/EditUserIT.java new file mode 100644 index 0000000..0291d02 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/cmd/EditUserIT.java @@ -0,0 +1,116 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.BaseIT; +import com.javarush.vasileva.entity.Role; +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.service.UserService; +import com.javarush.vasileva.util.Key; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.mockito.ArgumentCaptor; + +import java.util.Optional; + +import static com.javarush.vasileva.util.Link.USER_LIST; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNull; +import static org.mockito.Mockito.*; + +public class EditUserIT extends BaseIT { + + private final UserService userService = mock(UserService.class); + + private final EditUser editUser = new EditUser(userService); + + @Test + @DisplayName("When GET request with existing user then set user attribute") + void whenGetRequestWithExistingUser_ThenSetUserAttribute() { + String userIdStr = String.valueOf(testUser.getId()); + when(req.getParameter(Key.USER_ID)).thenReturn(userIdStr); + when(userService.getValidatedUser(userIdStr)).thenReturn(Optional.of(testUser)); + + String view = editUser.doGet(req); + + assertEquals(editUser.getView(), view); + verify(req).setAttribute(Key.USER, testUser); + verify(userService).getValidatedUser(userIdStr); + } + + @Test + @DisplayName("When GET request but user not found then do not set attribute") + void whenGetRequestButUserNotFound_ThenDoNotSetAttribute() { + String userIdStr = "999"; + when(req.getParameter(Key.USER_ID)).thenReturn(userIdStr); + when(userService.getValidatedUser(userIdStr)).thenReturn(Optional.empty()); + + String view = editUser.doGet(req); + + assertEquals(editUser.getView(), view); + verify(req, never()).setAttribute(eq(Key.USER), any()); + verify(userService).getValidatedUser(userIdStr); + } + + @Test + @DisplayName("When POST request to create user then call create and redirect") + void whenPostRequestToCreateUser_ThenCallCreateAndRedirect() { + when(req.getParameter(Key.CREATE)).thenReturn("true"); + when(req.getParameter(Key.LOGIN)).thenReturn(testUser.getLogin()); + when(req.getParameter(Key.EMAIL)).thenReturn(testUser.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(testUser.getPassword()); + when(req.getParameter(Key.ROLE)).thenReturn(Role.USER.toString()); + + String redirect = editUser.doPost(req); + + assertEquals(USER_LIST, redirect); + + ArgumentCaptor userCaptor = ArgumentCaptor.forClass(User.class); + verify(userService).create(userCaptor.capture()); + + User createdUser = userCaptor.getValue(); + assertEquals(testUser.getLogin(), createdUser.getLogin()); + assertEquals(testUser.getEmail(), createdUser.getEmail()); + assertEquals(testUser.getPassword(), createdUser.getPassword()); + assertEquals(testUser.getRole(), createdUser.getRole()); + assertNull(createdUser.getId()); + } + + @Test + @DisplayName("When POST request to update user then call update and redirect") + void whenPostRequestToUpdateUser_ThenCallUpdateAndRedirect() { + when(req.getParameter(Key.UPDATE)).thenReturn("true"); + when(req.getParameter(Key.USER_ID)).thenReturn(String.valueOf(testUser.getId())); + when(req.getParameter(Key.LOGIN)).thenReturn(testUser.getLogin()); + when(req.getParameter(Key.EMAIL)).thenReturn(testUser.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(testUser.getPassword()); + when(req.getParameter(Key.ROLE)).thenReturn(Role.USER.toString()); + + String redirect = editUser.doPost(req); + + assertEquals(USER_LIST, redirect); + + ArgumentCaptor userCaptor = ArgumentCaptor.forClass(User.class); + verify(userService).update(userCaptor.capture()); + + User updatedUser = userCaptor.getValue(); + assertEquals(testUser.getId(), updatedUser.getId()); + assertEquals(testUser.getLogin(), updatedUser.getLogin()); + assertEquals(testUser.getEmail(), updatedUser.getEmail()); + assertEquals(testUser.getPassword(), updatedUser.getPassword()); + assertEquals(Role.USER, updatedUser.getRole()); + } + + @Test + @DisplayName("When POST request without create/update params then do nothing and redirect") + void whenPostRequestWithoutCreateOrUpdate_ThenDoNothingAndRedirect() { + when(req.getParameter(Key.CREATE)).thenReturn(null); + when(req.getParameter(Key.UPDATE)).thenReturn(null); + when(req.getParameter(Key.ROLE)).thenReturn(Role.USER.toString()); + + String redirect = editUser.doPost(req); + + assertEquals(USER_LIST, redirect); + verify(userService, never()).create(any()); + verify(userService, never()).update(any()); + } + +} diff --git a/src/test/java/com/javarush/vasileva/cmd/HomeIT.java b/src/test/java/com/javarush/vasileva/cmd/HomeIT.java new file mode 100644 index 0000000..9c8074d --- /dev/null +++ b/src/test/java/com/javarush/vasileva/cmd/HomeIT.java @@ -0,0 +1,99 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.BaseIT; +import com.javarush.vasileva.entity.Quest; +import com.javarush.vasileva.entity.Role; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.AuthService; +import com.javarush.vasileva.service.QuestService; +import com.javarush.vasileva.util.Key; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.junit.jupiter.MockitoExtension; + +import java.util.Collections; +import java.util.List; +import java.util.Optional; + +import static com.javarush.vasileva.util.Key.QUEST_ID; +import static com.javarush.vasileva.util.Value.DELETE_QUEST_AUTH_ERROR; +import static com.javarush.vasileva.util.Value.QUEST_NOT_FOUND; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class HomeIT extends BaseIT { + + private final QuestService questService = mock(QuestService.class); + private final AuthService authService = mock(AuthService.class); + + private final Home home = new Home(questService, authService); + + @Test + @DisplayName("When GET request then retrieve all quests and set attribute") + void whenGetRequest_ThenRetrieveAllQuestsAndSetAttribute() { + List quests = Collections.singletonList(testQuest); + + when(questService.getAll()).thenReturn(quests); + + String view = home.doGet(req); + + assertEquals(home.getView(), view); + verify(req).setAttribute(Key.QUESTS, quests); + verify(questService).getAll(); + } + + @Test + @DisplayName("When DELETE request and admin authorized then delete quest and return view") + void whenDeleteRequestAndAdminAuthorized_ThenDeleteQuestAndReturnView() { + String questIdStr = String.valueOf(testQuest.getId()); + when(req.getParameter(QUEST_ID)).thenReturn(questIdStr); + when(questService.getValidatedQuest(questIdStr)).thenReturn(Optional.of(testQuest)); + when(session.getAttribute("role")).thenReturn(Role.ADMIN); + + doNothing().when(authService).checkAdminAuthorization(req, DELETE_QUEST_AUTH_ERROR); + + String view = home.doDelete(req); + + assertEquals(home.getView(), view); + verify(authService).checkAdminAuthorization(req, DELETE_QUEST_AUTH_ERROR); + verify(questService).getValidatedQuest(questIdStr); + verify(questService).delete(testQuest); + verify(req).setAttribute(Key.QUEST, testQuest); + } + + @Test + @DisplayName("When DELETE request but quest not found then throw IllegalArgumentException") + void whenDeleteRequestButQuestNotFound_ThenThrowIllegalArgumentException() { + String questIdStr = "999"; + when(req.getParameter(QUEST_ID)).thenReturn(questIdStr); + when(questService.getValidatedQuest(questIdStr)).thenReturn(Optional.empty()); + + doNothing().when(authService).checkAdminAuthorization(req, DELETE_QUEST_AUTH_ERROR); + + AppException exception = assertThrows(AppException.class, () -> home.doDelete(req)); + + assertTrue(exception.getMessage().contains(QUEST_NOT_FOUND + questIdStr)); + verify(authService).checkAdminAuthorization(req, DELETE_QUEST_AUTH_ERROR); + verify(questService).getValidatedQuest(questIdStr); + verifyNoMoreInteractions(questService); + } + + @Test + @DisplayName("When DELETE request but unauthorized then throw AppException") + void whenDeleteRequestButUnauthorized_ThenThrowAppException() { + String questIdStr = String.valueOf(testQuest.getId()); + when(req.getParameter(QUEST_ID)).thenReturn(questIdStr); + + doThrow(new AppException(DELETE_QUEST_AUTH_ERROR)) + .when(authService) + .checkAdminAuthorization(req, DELETE_QUEST_AUTH_ERROR); + + AppException exception = assertThrows(AppException.class, () -> home.doDelete(req)); + + assertEquals(DELETE_QUEST_AUTH_ERROR, exception.getMessage()); + verify(authService).checkAdminAuthorization(req, DELETE_QUEST_AUTH_ERROR); + verifyNoInteractions(questService); + } +} \ No newline at end of file diff --git a/src/test/java/com/javarush/vasileva/cmd/LoginIT.java b/src/test/java/com/javarush/vasileva/cmd/LoginIT.java new file mode 100644 index 0000000..e27a505 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/cmd/LoginIT.java @@ -0,0 +1,111 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.BaseIT; +import com.javarush.vasileva.entity.UserStats; +import com.javarush.vasileva.service.UserService; +import com.javarush.vasileva.service.UserStatsService; +import com.javarush.vasileva.util.Key; +import com.javarush.vasileva.util.Link; +import com.javarush.vasileva.util.Value; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.mockito.Mockito; + +import java.util.Optional; + +import static com.javarush.vasileva.util.Key.USER; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +public class LoginIT extends BaseIT { + + private final UserService userService = Mockito.mock(UserService.class); + private final UserStatsService userStatsService = Mockito.mock(UserStatsService.class); + + private final Login login = new Login(userService, userStatsService); + + @Test + @DisplayName("when valid credentials then redirect to HOME and set session attributes") + void whenValidCredentialsThenRedirectToHomeAndSetSessionAttributes() { + when(req.getParameter(Key.EMAIL)).thenReturn(testAdmin.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(testAdmin.getPassword()); + + when(userService.login(testAdmin.getEmail(), testAdmin.getPassword())) + .thenReturn(Optional.of(testAdmin)); + + UserStats stats = new UserStats(); + stats.setUserId(testAdmin.getId()); + when(userStatsService.getUserStats(testAdmin.getId())).thenReturn(Optional.of(stats)); + + String redirect = login.doPost(req); + + assertEquals(Link.HOME, redirect); + + verify(session).setAttribute(USER, testAdmin); + verify(session).setAttribute(Key.STATS, stats); + + verify(userService).login(testAdmin.getEmail(), testAdmin.getPassword()); + verify(userStatsService).getUserStats(testAdmin.getId()); + } + + @Test + @DisplayName("when invalid credentials then set error end return view") + void whenInvalidCredentials_ThenSetErrorEndReturnView() { + String invalidPassword = "wrongPassword"; + when(req.getParameter(Key.EMAIL)).thenReturn(testAdmin.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(invalidPassword); + + when(userService.login(testAdmin.getEmail(), invalidPassword)) + .thenReturn(Optional.empty()); + + String redirect = login.doPost(req); + + assertEquals(login.getView(), redirect); + verify(session).setAttribute(Key.ERROR, Value.INVALID_DATA_ERROR); + verify(userService).login(testAdmin.getEmail(), invalidPassword); + verifyNoMoreInteractions(userStatsService); + } + + @Test + @DisplayName("when empty email or password then set error and return view") + void whenEmptyEmailOrPassword_ThenSetErrorAndReturnView() { + String emptyEmail = ""; + String emptyPassword = ""; + when(req.getParameter(Key.EMAIL)).thenReturn(emptyEmail); + when(req.getParameter(Key.PASSWORD)).thenReturn(emptyPassword); + + String redirect = login.doPost(req); + + assertEquals(login.getView(), redirect); + verify(session).setAttribute(Key.ERROR, Value.EMPTY_DATA_ERROR); + verifyNoInteractions(userService); + verifyNoInteractions(userStatsService); + } + + @Test + @DisplayName("when user has no stats then create new stats") + void whenUserHasNoStats_ThenCreateNewStats() { + when(req.getParameter(Key.EMAIL)).thenReturn(testAdmin.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(testAdmin.getPassword()); + + when(userService.login(testAdmin.getEmail(), testAdmin.getPassword())) + .thenReturn(Optional.of(testAdmin)); + + UserStats stats = new UserStats(); + stats.setUserId(testAdmin.getId()); + when(userStatsService.getUserStats(testAdmin.getId())) + .thenReturn(Optional.empty()); + when(userStatsService.createUserStats(testAdmin.getId())) + .thenReturn(stats); + + String redirect = login.doPost(req); + + assertEquals(Link.HOME, redirect); + verify(session).setAttribute(Key.USER, testAdmin); + verify(session).setAttribute(Key.STATS, stats); + verify(userStatsService).createUserStats(testAdmin.getId()); + + } + + +} diff --git a/src/test/java/com/javarush/vasileva/cmd/LogoutIT.java b/src/test/java/com/javarush/vasileva/cmd/LogoutIT.java new file mode 100644 index 0000000..018966b --- /dev/null +++ b/src/test/java/com/javarush/vasileva/cmd/LogoutIT.java @@ -0,0 +1,40 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.BaseIT; +import com.javarush.vasileva.util.Link; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; + +import static com.javarush.vasileva.util.Link.LOGIN; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.*; + +public class LogoutIT extends BaseIT { + + private final Logout logout = new Logout(); + + @Test + @DisplayName("when session exists then invalidate it and redirect to LOGIN") + void whenSessionExists_ThenInvalidateAndRedirect() { + when(req.getSession(false)).thenReturn(session); + when(session.getId()).thenReturn("test-session-id"); + + String redirect = logout.doGet(req); + + assertEquals(Link.LOGIN, redirect); + verify(req).getSession(false); + verify(session).invalidate(); + } + + @Test + @DisplayName("when no session exists then skip invalidation and redirect to LOGIN") + void whenNoSessionExists_ThenSkipInvalidationAndRedirect() { + when(req.getSession(false)).thenReturn(null); + + String redirect = logout.doGet(req); + + assertEquals(LOGIN, redirect); + verify(req).getSession(false); + verify(session, never()).invalidate(); + } +} diff --git a/src/test/java/com/javarush/vasileva/cmd/PlayGameIT.java b/src/test/java/com/javarush/vasileva/cmd/PlayGameIT.java new file mode 100644 index 0000000..36f1f1a --- /dev/null +++ b/src/test/java/com/javarush/vasileva/cmd/PlayGameIT.java @@ -0,0 +1,116 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.BaseIT; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.GameService; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; + +import java.util.Optional; + +import static com.javarush.vasileva.util.Key.*; +import static com.javarush.vasileva.util.Value.*; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.mockito.Mockito.*; + +public class PlayGameIT extends BaseIT { + + private final GameService gameService = mock(GameService.class); + + private final PlayGame playGame = new PlayGame(gameService); + + @Test + @DisplayName("When GET request with new game then start new game and set attributes") + void whenGetRequestWithNewGame_ThenStartNewGameAndSetAttributes() { + Long questId = testQuest.getId(); + + when(req.getParameter(QUEST_ID)).thenReturn(String.valueOf(questId)); + when(req.getParameter(GAME_ID)).thenReturn(null); + + when(session.getAttribute(USER)).thenReturn(testUser); + when(gameService.startNewGame(questId, testUser.getId())).thenReturn(testGame); + + String view = playGame.doGet(req); + + assertEquals(playGame.getView(), view); + + verify(req).setAttribute(GAME, testGame); + verify(req).setAttribute(STATE, testGame.getGameState().isCompleted()); + verify(req).setAttribute(QUEST, testGame.getGameState().getCurrentQuest()); + verify(req).setAttribute(WINNING, + testGame.getGameState().getCurrentQuestion().getLabel().contains(WIN)); + + verify(gameService).startNewGame(questId, testUser.getId()); + } + + @Test + @DisplayName("When GET request with existing game then retrieve game and set attributes") + void whenGetRequestWithExistingGame_ThenRetrieveGameAndSetAttributes() { + when(req.getParameter(QUEST_ID)).thenReturn(String.valueOf(testQuest.getId())); + when(req.getParameter(GAME_ID)).thenReturn(String.valueOf(testGame.getId())); + + when(session.getAttribute(USER)).thenReturn(testUser); + when(gameService.getGameById(testGame.getId())).thenReturn(Optional.of(testGame)); + + String view = playGame.doGet(req); + + assertEquals(playGame.getView(), view); + verify(req).setAttribute(GAME, testGame); + verify(req).setAttribute(STATE, testGame.getGameState().isCompleted()); + verify(req).setAttribute(QUEST, testGame.getGameState().getCurrentQuest()); + verify(req).setAttribute(WINNING, + testGame.getGameState().getCurrentQuestion().getLabel().contains(WIN)); + verify(gameService).getGameById(testGame.getId()); + } + + @Test + @DisplayName("When GET request but user not authenticated then throw AppException") + void whenGetRequestButUserNotAuthenticated_ThenThrowAppException() { + when(req.getParameter(QUEST_ID)).thenReturn(String.valueOf(testQuest.getId())); + when(req.getParameter(GAME_ID)).thenReturn(null); + + when(session.getAttribute(USER)).thenReturn(null); + + AppException exception = assertThrows(AppException.class, () -> playGame.doGet(req)); + + assertEquals(AUTH_ERROR, exception.getMessage()); + verify(gameService, never()).startNewGame(anyLong(), anyLong()); + verify(gameService, never()).getGameById(anyLong()); + } + + @Test + @DisplayName("When GET request but game not found then throw AppException") + void whenGetRequestButGameNotFound_ThenThrowAppException() { + Long gameId = 999L; + when(req.getParameter(QUEST_ID)).thenReturn(String.valueOf(testQuest.getId())); + when(req.getParameter(GAME_ID)).thenReturn(String.valueOf(gameId)); + + when(session.getAttribute(USER)).thenReturn(testUser); + when(gameService.getGameById(gameId)).thenReturn(Optional.empty()); + + AppException exception = assertThrows(AppException.class, () -> playGame.doGet(req)); + + assertEquals(GAME_NOT_FOUND, exception.getMessage()); + verify(gameService).getGameById(gameId); + } + + @Test + @DisplayName("When POST request then advance game and return redirect URL") + void whenPostRequest_ThenAdvanceGameAndReturnRedirectUrl() throws AppException { + when(req.getParameter(GAME_ID)).thenReturn(String.valueOf(testGame.getId())); + when(req.getParameter(SELECTED_ANSWER_ID)).thenReturn(String.valueOf(testAnswer1.getId())); + + when(session.getAttribute(USER)).thenReturn(testUser); + when(gameService.advanceGame(testGame.getId(), testAnswer1.getId())).thenReturn(testGame); + + String redirectUrl = playGame.doPost(req); + String expectedUrl = playGame.getView() + "?" + GAME_ID + "=" + testGame.getId() + + "&" + QUEST_ID + "=" + testGame.getGameState().getCurrentQuest().getId(); + + assertEquals(expectedUrl, redirectUrl); + verify(req).setAttribute(GAME, testGame); + verify(req).setAttribute(USER, testUser); + verify(gameService).advanceGame(testGame.getId(), testAnswer1.getId()); + } +} diff --git a/src/test/java/com/javarush/vasileva/cmd/RegisterIT.java b/src/test/java/com/javarush/vasileva/cmd/RegisterIT.java new file mode 100644 index 0000000..12a8daa --- /dev/null +++ b/src/test/java/com/javarush/vasileva/cmd/RegisterIT.java @@ -0,0 +1,116 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.BaseIT; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.UserService; +import com.javarush.vasileva.util.Key; +import com.javarush.vasileva.util.Link; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; + +import static com.javarush.vasileva.util.Value.EMPTY_DATA_ERROR; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.mockito.Mockito.*; + +public class RegisterIT extends BaseIT { + + private final UserService userService = mock(UserService.class); + + private final Register register = new Register(userService); + + @Test + @DisplayName("when valid data then register user and redirect to login") + void whenValidData_ThenRegisterUserAndRedirectToLogin() { + + when(req.getParameter(com.javarush.vasileva.util.Key.LOGIN)).thenReturn(testGuest.getLogin()); + when(req.getParameter(Key.EMAIL)).thenReturn(testGuest.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(testGuest.getPassword()); + + String redirect = register.doPost(req); + + assertEquals(Link.LOGIN, redirect); + verify(userService).register(testGuest.getLogin(), testGuest.getEmail(), testGuest.getPassword()); + } + + @Test + @DisplayName("when empty login then throw AppException") + void whenEmptyLogin_ThenThrowAppException() { + String emptyLogin = ""; + when(req.getParameter(Key.LOGIN)).thenReturn(emptyLogin); + when(req.getParameter(Key.EMAIL)).thenReturn(testGuest.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(testGuest.getPassword()); + + AppException exception = assertThrows(AppException.class, () -> register.doPost(req)); + + assertEquals(EMPTY_DATA_ERROR, exception.getMessage()); + verifyNoInteractions(userService); + } + + @Test + @DisplayName("when empty email then throw AppException") + void whenEmptyEmailThenThrowAppException() { + String emptyEmail = ""; + when(req.getParameter(Key.LOGIN)).thenReturn(testGuest.getLogin()); + when(req.getParameter(Key.EMAIL)).thenReturn(emptyEmail); + when(req.getParameter(Key.PASSWORD)).thenReturn(testGuest.getPassword()); + + AppException exception = assertThrows(AppException.class, () -> register.doPost(req)); + + assertEquals(EMPTY_DATA_ERROR, exception.getMessage()); + verifyNoInteractions(userService); + } + + @Test + @DisplayName("when empty password then throw AppException") + void whenEmptyPasswordThenThrowAppException() { + String emptyPassword = ""; + when(req.getParameter(Key.LOGIN)).thenReturn(testGuest.getLogin()); + when(req.getParameter(Key.EMAIL)).thenReturn(testGuest.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(emptyPassword); + + AppException exception = assertThrows(AppException.class, () -> register.doPost(req)); + + assertEquals(EMPTY_DATA_ERROR, exception.getMessage()); + verifyNoInteractions(userService); + } + + @Test + @DisplayName("when null login then throw AppException") + void whenNullLogin_ThenThrowAppException() { + when(req.getParameter(Key.LOGIN)).thenReturn(null); + when(req.getParameter(Key.EMAIL)).thenReturn(testGuest.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(testGuest.getPassword()); + + AppException exception = assertThrows(AppException.class, () -> register.doPost(req)); + + assertEquals(EMPTY_DATA_ERROR, exception.getMessage()); + verifyNoInteractions(userService); + } + + @Test + @DisplayName("when null email then throw AppException") + void whenNullEmail_ThenThrowAppException() { + when(req.getParameter(Key.LOGIN)).thenReturn(testGuest.getLogin()); + when(req.getParameter(Key.EMAIL)).thenReturn(null); + when(req.getParameter(Key.PASSWORD)).thenReturn(testGuest.getPassword()); + + AppException exception = assertThrows(AppException.class, () -> register.doPost(req)); + + assertEquals(EMPTY_DATA_ERROR, exception.getMessage()); + verifyNoInteractions(userService); + } + + @Test + @DisplayName("when null password then throw AppException") + void whenNullPasswordThenThrowAppException() { + when(req.getParameter(Key.LOGIN)).thenReturn(testGuest.getLogin()); + when(req.getParameter(Key.EMAIL)).thenReturn(testGuest.getEmail()); + when(req.getParameter(Key.PASSWORD)).thenReturn(null); + + AppException exception = assertThrows(AppException.class, () -> register.doPost(req)); + + assertEquals(EMPTY_DATA_ERROR, exception.getMessage()); + verifyNoInteractions(userService); + } +} diff --git a/src/test/java/com/javarush/vasileva/cmd/StatsIT.java b/src/test/java/com/javarush/vasileva/cmd/StatsIT.java new file mode 100644 index 0000000..6b35913 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/cmd/StatsIT.java @@ -0,0 +1,68 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.BaseIT; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.UserStatsService; +import com.javarush.vasileva.util.Key; +import com.javarush.vasileva.util.Value; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; + +import java.util.Optional; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.mockito.Mockito.*; + +public class StatsIT extends BaseIT { + + private final UserStatsService statsService = mock(UserStatsService.class); + + private final Stats stats = new Stats(statsService); + + @Test + @DisplayName("when user is authenticated then load stats and return view") + void whenUserAuthenticated_ThenLoadStatsAndReturnView() { + when(session.getAttribute(Key.USER)).thenReturn(testUser); + when(statsService.getUserStats(testUser.getId())).thenReturn(Optional.of(testUserStats)); + + String view = stats.doGet(req); + + assertEquals(stats.getView(), view); + + verify(req).getSession(); + verify(session).getAttribute(Key.USER); + verify(statsService).getUserStats(testUser.getId()); + verify(req).setAttribute(eq(Key.STATS), eq(testUserStats)); + } + + @Test + @DisplayName("when user not authenticated then throw AppException") + void whenUserNotAuthenticated_ThenThrowAppException() { + when(session.getAttribute(Key.USER)).thenReturn(null); + + AppException exception = assertThrows(AppException.class, () -> stats.doGet(req)); + + assertEquals(Value.AUTH_ERROR, exception.getMessage()); + verify(req).getSession(); + verify(session).getAttribute(Key.USER); + verify(statsService, never()).getUserStats(anyLong()); + } + + @Test + @DisplayName("when stats not found then throw AppException") + void whenStatsNotFound_ThenThrowAppException() { + when(session.getAttribute(Key.USER)).thenReturn(testUser); + when(statsService.getUserStats(testUser.getId())).thenReturn(Optional.empty()); + + AppException exception = assertThrows(AppException.class, () -> stats.doGet(req)); + + assertEquals(Value.STATS_NOT_FOUND, exception.getMessage()); + + verify(req).getSession(); + verify(session).getAttribute(Key.USER); + verify(statsService).getUserStats(testUser.getId()); + verify(req, never()).setAttribute(eq(Key.STATS), any()); + } + +} diff --git a/src/test/java/com/javarush/vasileva/cmd/UserListIT.java b/src/test/java/com/javarush/vasileva/cmd/UserListIT.java new file mode 100644 index 0000000..10064e7 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/cmd/UserListIT.java @@ -0,0 +1,99 @@ +package com.javarush.vasileva.cmd; + +import com.javarush.vasileva.BaseIT; +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.service.AuthService; +import com.javarush.vasileva.service.UserService; +import com.javarush.vasileva.util.Key; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; + +import java.util.Arrays; +import java.util.List; +import java.util.Optional; + +import static com.javarush.vasileva.util.Key.USER_ID; +import static com.javarush.vasileva.util.Value.USER_LIST_AUTH_ERROR; +import static com.javarush.vasileva.util.Value.USER_NOT_FOUND; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +public class UserListIT extends BaseIT { + + private final UserService userService = mock(UserService.class); + private final AuthService authService = mock(AuthService.class); + + private final UserList userList = new UserList(userService, authService); + + @Test + @DisplayName("When GET request and admin authorized then retrieve users and set attributes") + void whenGetRequestAndAdminAuthorized_ThenRetrieveUsersAndSetAttributes() { + List users = Arrays.asList(testAdmin, testUser, testGuest); + when(userService.getAll()).thenReturn(users); + + doNothing().when(authService).checkAdminAuthorization(req, USER_LIST_AUTH_ERROR); + + String view = userList.doGet(req); + assertEquals(userList.getView(), view); + verify(authService).checkAdminAuthorization(req, USER_LIST_AUTH_ERROR); + verify(userService).getAll(); + verify(req).setAttribute(Key.USERS, users); + } + + @Test + @DisplayName("When GET request but unauthorized then throw AppException") + void whenGetRequestButUnauthorized_ThenThrowAppException() { + doThrow(new AppException(USER_LIST_AUTH_ERROR)) + .when(authService) + .checkAdminAuthorization(req, USER_LIST_AUTH_ERROR); + + AppException exception = assertThrows(AppException.class, () -> userList.doGet(req)); + + assertEquals(USER_LIST_AUTH_ERROR, exception.getMessage()); + verify(authService).checkAdminAuthorization(req, USER_LIST_AUTH_ERROR); + verifyNoInteractions(userService); + } + + @Test + @DisplayName("When DELETE request and user exists then delete user and return view") + void whenDeleteRequestAndUserExistsThenDeleteUserAndReturnView() { + String userIdStr = String.valueOf(testUser.getId()); + + when(req.getParameter(USER_ID)).thenReturn(userIdStr); + when(userService.getValidatedUser(userIdStr)).thenReturn(Optional.of(testUser)); + + String view = userList.doDelete(req); + + assertEquals(userList.getView(), view); + + verify(userService).getValidatedUser(userIdStr); + verify(userService).delete(testUser); + } + + @Test + @DisplayName("When DELETE request but user not found then throw AppException") + void whenDeleteRequestButUserNotFound_ThenThrowAppException() { + String userIdStr = "999"; + when(req.getParameter(USER_ID)).thenReturn(userIdStr); + when(userService.getValidatedUser(userIdStr)).thenReturn(Optional.empty()); + + AppException exception = assertThrows(AppException.class, () -> userList.doDelete(req)); + + assertEquals(USER_NOT_FOUND + userIdStr, exception.getMessage()); + + verify(userService).getValidatedUser(userIdStr); + verifyNoMoreInteractions(userService); + } + + @Test + @DisplayName("When DELETE request with null userId then throw AppException") + void whenDeleteRequestWithNullUserId_ThenThrowAppException() { + when(req.getParameter(USER_ID)).thenReturn(null); + AppException exception = assertThrows(AppException.class, () -> userList.doDelete(req)); + + assertTrue(exception.getMessage().startsWith(USER_NOT_FOUND)); + verify(userService).getValidatedUser(null); + verifyNoMoreInteractions(userService); + } +} diff --git a/src/test/java/com/javarush/vasileva/service/AnswerServiceTest.java b/src/test/java/com/javarush/vasileva/service/AnswerServiceTest.java new file mode 100644 index 0000000..7fe3faa --- /dev/null +++ b/src/test/java/com/javarush/vasileva/service/AnswerServiceTest.java @@ -0,0 +1,73 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Answer; +import com.javarush.vasileva.repository.AnswerRepository; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; + +import java.util.List; +import java.util.Optional; + +import static com.javarush.vasileva.service.TestData.*; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.verify; +import static org.mockito.Mockito.when; + +@ExtendWith(MockitoExtension.class) +public class AnswerServiceTest { + @Mock + private AnswerRepository answerRepository; + + @InjectMocks + private AnswerService answerService; + + @Test + @DisplayName("when getAll() then return all answers from repository") + void testGetAll() { + List answers = createMultipleAnswers(); + when(answerRepository.getAll()).thenReturn(answers); + + List result = answerService.getAll(); + + assertEquals(answers, result); + verify(answerRepository).getAll(); + } + + @Test + @DisplayName("when findById() then return answer by ID if it exists") + void whenFindById_thenExisting() { + Answer expected = createValidAnswer(); + when(answerRepository.findById(VALID_ANSWER_ID)) + .thenReturn(Optional.of(expected)); + + Optional result = answerService.findById(VALID_ANSWER_ID); + + assertTrue(result.isPresent()); + assertEquals(expected, result.get()); + verify(answerRepository).findById(VALID_ANSWER_ID); + } + + @Test + @DisplayName("when findById() then return empty Optional if answer is not found") + void whenFindById_ThenNonExisting() { + when(answerRepository.findById(NON_EXISTENT_ANSWER_ID)) + .thenReturn(Optional.empty()); + + Optional result = answerService.findById(NON_EXISTENT_ANSWER_ID); + + assertFalse(result.isPresent()); + verify(answerRepository).findById(NON_EXISTENT_ANSWER_ID); + } + + @Test + @DisplayName("when create() then delegate saving answer to repository") + void testCreate() { + Answer newAnswer = createValidAnswer(); + answerService.create(newAnswer); + verify(answerRepository).create(newAnswer); + } +} diff --git a/src/test/java/com/javarush/vasileva/service/AuthServiceTest.java b/src/test/java/com/javarush/vasileva/service/AuthServiceTest.java new file mode 100644 index 0000000..df87921 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/service/AuthServiceTest.java @@ -0,0 +1,86 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.exception.AppException; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpSession; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; + +import static com.javarush.vasileva.service.TestData.*; +import static com.javarush.vasileva.util.Key.USER; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class AuthServiceTest { + + @Mock + private HttpServletRequest request; + @Mock + private HttpSession session; + + @InjectMocks + private AuthService authService; + + @Test + @DisplayName("checkAdminAuthorization() должен пройти успешно для пользователя с ролью ADMIN") + void testCheckAdminAuthorization_AdminSuccess() { + User adminUser = createAdminUser(); + when(request.getSession(false)).thenReturn(session); + when(session.getAttribute(USER)).thenReturn(adminUser); + + assertDoesNotThrow(() -> authService.checkAdminAuthorization(request, UNAUTHORIZED_MESSAGE)); + verify(request).getSession(false); + verify(session).getAttribute(USER); + } + + @Test + @DisplayName("checkAdminAuthorization() должен выбросить AppException для null-пользователя") + void testCheckAdminAuthorization_NullUserThrowsException() { + when(request.getSession(false)).thenReturn(session); + when(session.getAttribute(USER)).thenReturn(createNullUser()); + + AppException exception = assertThrows(AppException.class, + () -> authService.checkAdminAuthorization(request, UNAUTHORIZED_MESSAGE)); + + assertEquals(UNAUTHORIZED_MESSAGE, exception.getMessage()); + verify(request).getSession(false); + verify(session).getAttribute(USER); + } + + @Test + @DisplayName("checkAdminAuthorization() должен выбросить AppException для пользователя с ролью USER") + void testCheckAdminAuthorization_UserRoleThrowsException() { + User userUser = createUserUser(); + when(request.getSession(false)).thenReturn(session); + when(session.getAttribute(USER)).thenReturn(userUser); + + AppException exception = assertThrows(AppException.class, + () -> authService.checkAdminAuthorization(request, UNAUTHORIZED_MESSAGE)); + + assertEquals(UNAUTHORIZED_MESSAGE, exception.getMessage()); + verify(request).getSession(false); + verify(session).getAttribute(USER); + } + + @Test + @DisplayName("checkAdminAuthorization() должен выбросить AppException для пользователя с ролью GUEST") + void testCheckAdminAuthorization_GuestRoleThrowsException() { + User guestUser = createGuestUser(); + when(request.getSession(false)).thenReturn(session); + when(session.getAttribute(USER)).thenReturn(guestUser); + + AppException exception = assertThrows(AppException.class, + () -> authService.checkAdminAuthorization(request, UNAUTHORIZED_MESSAGE)); + + assertEquals(UNAUTHORIZED_MESSAGE, exception.getMessage()); + verify(request).getSession(false); + verify(session).getAttribute(USER); + } + +} diff --git a/src/test/java/com/javarush/vasileva/service/GameServiceTest.java b/src/test/java/com/javarush/vasileva/service/GameServiceTest.java new file mode 100644 index 0000000..ace7cd5 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/service/GameServiceTest.java @@ -0,0 +1,163 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Game; +import com.javarush.vasileva.entity.GameState; +import com.javarush.vasileva.entity.Quest; +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.exception.AppException; +import com.javarush.vasileva.game.GameEngine; +import com.javarush.vasileva.repository.InMemoryGameRepository; +import com.javarush.vasileva.util.Value; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; + +import java.util.Optional; + +import static com.javarush.vasileva.service.TestData.*; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class GameServiceTest { + @Mock + private InMemoryGameRepository gameRepository; + @Mock + private UserService userService; + @Mock + private QuestService questService; + @Mock + private GameEngine gameEngine; + + @InjectMocks + private GameService gameService; + + private User user; + + @BeforeEach + public void setUp() { + user = createValidUser(); + } + + @Test + @DisplayName("when startNewGame() then create and save new game when user and quest exist") + void whenStartNewGame_ThenSuccess() { + Quest quest = createValidQuest(); + GameState initialState = createInitialGameState(); + Game expectedGame = createGame( + VALID_QUEST_ID, + VALID_USER_ID, + initialState.getCurrentQuestion().getGeneratedId(), + initialState + ); + + when(userService.findById(VALID_USER_ID)).thenReturn(Optional.of(user)); + when(questService.findById(VALID_QUEST_ID)).thenReturn(Optional.of(quest)); + when(gameEngine.startGame(user, quest)).thenReturn(initialState); + when(gameRepository.save(any(Game.class))).thenReturn(expectedGame); + + Game result = gameService.startNewGame(VALID_QUEST_ID, VALID_USER_ID); + + assertEquals(expectedGame, result); + verify(userService).findById(VALID_USER_ID); + verify(questService).findById(VALID_QUEST_ID); + verify(gameEngine).startGame(user, quest); + verify(gameRepository).save(expectedGame); + } + + @Test + @DisplayName("when startNewGame() then throw AppException if user not found") + void whenStartNewGame_ThenUserNotFound() { + when(userService.findById(NON_EXISTENT_USER_ID)) + .thenReturn(Optional.empty()); + + AppException exception = assertThrows(AppException.class, + () -> gameService.startNewGame(VALID_QUEST_ID, NON_EXISTENT_USER_ID)); + + assertEquals(Value.USER_NOT_FOUND, exception.getMessage()); + verify(userService).findById(NON_EXISTENT_USER_ID); + verify(questService, never()).findById(anyLong()); + verify(gameEngine, never()).startGame(any(), any()); + verify(gameRepository, never()).save(any()); + } + + @Test + @DisplayName("when startNewGame() then throw AppException if quest not found") + void whenStartNewGame_ThenQuestNotFound() { + when(userService.findById(VALID_USER_ID)).thenReturn(Optional.of(user)); + when(questService.findById(NON_EXISTENT_QUEST_ID)) + .thenReturn(Optional.empty()); + + AppException exception = assertThrows(AppException.class, + () -> gameService.startNewGame(NON_EXISTENT_QUEST_ID, VALID_USER_ID)); + + assertEquals(Value.QUEST_NOT_FOUND, exception.getMessage()); + verify(userService).findById(VALID_USER_ID); + verify(questService).findById(NON_EXISTENT_QUEST_ID); + verify(gameEngine, never()).startGame(any(), any()); + verify(gameRepository, never()).save(any()); + } + + @Test + @DisplayName("advanceGame() should update game state when game exists") + void testAdvanceGame_Success() { + Game existingGame = createSavedGame(); + GameState nextState = createNextGameState(); + + when(gameRepository.findById(VALID_GAME_ID)) + .thenReturn(Optional.of(existingGame)); + when(gameEngine.advanceGame(existingGame.getGameState(), VALID_ANSWER_ID)) + .thenReturn(nextState); + when(gameRepository.save(any(Game.class))).thenAnswer(invocation -> { + Game updatedGame = invocation.getArgument(0); + updatedGame.setId(VALID_GAME_ID); + return updatedGame; + }); + + Game result = gameService.advanceGame(VALID_GAME_ID, VALID_ANSWER_ID); + + assertNotNull(result); + assertEquals(VALID_GAME_ID, result.getId()); + assertEquals(nextState.getCurrentQuestion().getGeneratedId(), result.getCurrentQuestionId()); + assertEquals(nextState, result.getGameState()); + + verify(gameEngine).advanceGame( + argThat(state -> state.getCurrentQuestion().getGeneratedId().equals(1L)), + eq(TestData.VALID_ANSWER_ID) + ); + verify(gameRepository).save(result); + } + + @Test + @DisplayName("when getGameById() then return game if exists") + void testGetGameById_Found() { + Game expectedGame = createSavedGame(); + when(gameRepository.findById(VALID_GAME_ID)) + .thenReturn(Optional.of(expectedGame)); + + Optional result = gameService.getGameById(VALID_GAME_ID); + + assertTrue(result.isPresent()); + assertEquals(expectedGame, result.get()); + verify(gameRepository).findById(VALID_GAME_ID); + } + + @Test + @DisplayName("when getGameById() then return empty Optional if game not found") + void whenGetGameById_ThenNotFound() { + when(gameRepository.findById(NON_EXISTENT_GAME_ID)) + .thenReturn(Optional.empty()); + + Optional result = gameService.getGameById(NON_EXISTENT_GAME_ID); + + assertFalse(result.isPresent()); + verify(gameRepository).findById(NON_EXISTENT_GAME_ID); + } + + +} diff --git a/src/test/java/com/javarush/vasileva/service/QuestServiceTest.java b/src/test/java/com/javarush/vasileva/service/QuestServiceTest.java new file mode 100644 index 0000000..9dbd094 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/service/QuestServiceTest.java @@ -0,0 +1,138 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Quest; +import com.javarush.vasileva.repository.QuestRepository; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; + +import java.util.List; +import java.util.Optional; + +import static com.javarush.vasileva.service.TestData.*; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class QuestServiceTest { + + @Mock + private QuestRepository questRepository; + + @InjectMocks + private QuestService questService; + + private Quest quest; + + @BeforeEach + public void setUp() { + quest = createValidQuest(); + } + + @Test + @DisplayName("when getAll() then return all quests from repository") + void whenGetAll_thenReturnAllQuestsFromRepository() { + List quests = createMultipleQuests(); + when(questRepository.getAll()).thenReturn(quests); + + List result = questService.getAll(); + + assertEquals(quests, result); + verify(questRepository).getAll(); + } + + @Test + @DisplayName("when findById() then return quest by ID") + void whenFindById_ThenFindById() { + long questId = quest.getId(); + when(questRepository.findById(questId)).thenReturn(Optional.of(quest)); + + Optional result = questService.findById(questId); + + assertTrue(result.isPresent()); + assertEquals(quest, result.get()); + verify(questRepository).findById(questId); + } + + @Test + @DisplayName("when findById() then return empty Optional if quest not found") + void whenFindById_ThenNotFound() { + long questId = NON_EXISTENT_QUEST_ID; + when(questRepository.findById(questId)).thenReturn(Optional.empty()); + + Optional result = questService.findById(questId); + + assertFalse(result.isPresent()); + verify(questRepository).findById(questId); + } + + @Test + @DisplayName("when create() then delegate to repository") + void testCreate() { + questService.create(quest); + verify(questRepository).create(quest); + } + + @Test + @DisplayName("when update() than delegate to repository") + void testUpdate() { + questService.update(quest); + verify(questRepository).update(quest); + } + + @Test + @DisplayName("when delete() then delegate to repository") + void testDelete() { + questService.delete(quest); + verify(questRepository).delete(quest); + } + + @Test + @DisplayName("when getValidatedQuest() then return empty when questIdStr is null") + void givenQuestIdNull_whenGetValidatedQuest_ThenReturnEmpty() { + Optional result = questService.getValidatedQuest(NULL_QUEST_ID_STR); + assertFalse(result.isPresent()); + verify(questRepository, never()).findById(anyLong()); + } + + @Test + @DisplayName("when getValidatedQuest() then return empty when questIdStr is empty") + void givenEmptyQuestId_WhenGetValidatedQuest_ThenReturnEmpty() { + Optional result = questService.getValidatedQuest(EMPTY_QUEST_ID_STR); + assertFalse(result.isPresent()); + verify(questRepository, never()).findById(anyLong()); + } + + @Test + @DisplayName("when getValidatedQuest() then parse string to long and find quest") + void whenGetValidatedQuest_ThenReturnQuest() { + String questIdStr = String.valueOf(quest.getId()); + long questId = Long.parseLong(questIdStr); + + when(questRepository.findById(questId)).thenReturn(Optional.of(quest)); + + Optional result = questService.getValidatedQuest(questIdStr); + + assertTrue(result.isPresent()); + assertEquals(quest, result.get()); + verify(questRepository).findById(questId); + } + + @Test + @DisplayName("when getValidatedQuest() then return empty if quest not found by parsed ID") + void whenGetValidatedQuest_thenQuestNotFound() { + String questIdStr = NON_EXISTENT_QUEST_ID_STR; + long questId = Long.parseLong(questIdStr); + + when(questRepository.findById(questId)).thenReturn(Optional.empty()); + + Optional result = questService.getValidatedQuest(questIdStr); + + assertFalse(result.isPresent()); + verify(questRepository).findById(questId); + } +} diff --git a/src/test/java/com/javarush/vasileva/service/QuestionServiceTest.java b/src/test/java/com/javarush/vasileva/service/QuestionServiceTest.java new file mode 100644 index 0000000..a7020ff --- /dev/null +++ b/src/test/java/com/javarush/vasileva/service/QuestionServiceTest.java @@ -0,0 +1,139 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Question; +import com.javarush.vasileva.repository.QuestionRepository; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; + +import java.util.List; +import java.util.Optional; + +import static com.javarush.vasileva.service.TestData.*; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.verify; +import static org.mockito.Mockito.when; + +@ExtendWith(MockitoExtension.class) +public class QuestionServiceTest { + + @Mock + private QuestionRepository questionRepository; + + @InjectMocks + private QuestionService questionService; + + @Test + @DisplayName("when create() then delegate to repository") + void testCreate() { + Question question = createSimpleQuestion(); + questionService.create(question); + verify(questionRepository).create(question); + } + + @Test + @DisplayName("when getAll() then return all questions from repository") + void testGetAll() { + List questions = createMultipleQuestions(); + when(questionRepository.getAll()).thenReturn(questions); + + List result = questionService.getAll(); + + assertEquals(questions, result); + verify(questionRepository).getAll(); + } + + @Test + @DisplayName("when findById() then return question by ID") + void testFindById() { + Question expected = createQuestionWithAnswers(); + when(questionRepository.findById(VALID_QUESTION_ID)) + .thenReturn(Optional.of(expected)); + + Optional result = questionService.findById(VALID_QUESTION_ID); + + assertTrue(result.isPresent()); + assertEquals(expected, result.get()); + verify(questionRepository).findById(VALID_QUESTION_ID); + } + + @Test + @DisplayName("when findById() then return empty Optional if question not found") + void whenFindById_ThenQuestionNotFound() { + when(questionRepository.findById(NON_EXISTENT_QUESTION_ID)) + .thenReturn(Optional.empty()); + + Optional result = questionService.findById(NON_EXISTENT_QUESTION_ID); + + assertFalse(result.isPresent()); + verify(questionRepository).findById(NON_EXISTENT_QUESTION_ID); + } + + @Test + @DisplayName("when getByQuestionLabelAndQuestId() then return question by label and quest ID") + void testGetByQuestionLabelAndQuestId() { + Question expected = createQuestionWithAnswers(); + when(questionRepository.getByQuestionLabelAndQuestId( + VALID_LABEL, VALID_QUEST_ID)) + .thenReturn(Optional.of(expected)); + + Optional result = questionService.getByQuestionLabelAndQuestId( + VALID_LABEL, VALID_QUEST_ID); + + assertTrue(result.isPresent()); + assertEquals(expected, result.get()); + verify(questionRepository).getByQuestionLabelAndQuestId( + VALID_LABEL, VALID_QUEST_ID); + } + + @Test + @DisplayName("when getByQuestionLabelAndQuestId() then return empty if not found") + void whenGetByQuestionLabelAndQuestId_ThenQuestionNotFound() { + when(questionRepository.getByQuestionLabelAndQuestId( + INVALID_LABEL, VALID_QUEST_ID)) + .thenReturn(Optional.empty()); + + Optional result = questionService.getByQuestionLabelAndQuestId( + INVALID_LABEL, VALID_QUEST_ID); + + assertFalse(result.isPresent()); + verify(questionRepository).getByQuestionLabelAndQuestId( + INVALID_LABEL, VALID_QUEST_ID); + } + + @Test + @DisplayName("when isFinalQuestion() then return true for question with null answers") + void whenIsFinalQuestion_ThenNullAnswers() { + Question question = createFinalQuestion(); + + boolean result = questionService.isFinalQuestion(question); + + assertTrue(result); + } + + @Test + @DisplayName("when isFinalQuestion() then return true for question with empty answers list") + void whenIsFinalQuestion_EmptyAnswers() { + Question question = createSimpleQuestion(); + question.setAnswers(List.of()); + + boolean result = questionService.isFinalQuestion(question); + + assertTrue(result, "Question is final if answer list is empty"); + assertEquals(List.of(), question.getAnswers()); + } + + @Test + @DisplayName("when isFinalQuestion() then return false for question with non‑empty answers list") + void whenIsNotFinalQuestion_thenNonEmptyAnswers() { + Question question = createQuestionWithAnswers(); + assertFalse(question.getAnswers().isEmpty()); + + boolean result = questionService.isFinalQuestion(question); + + assertFalse(result, "Question is not final if it has answers"); + } +} diff --git a/src/test/java/com/javarush/vasileva/service/TestData.java b/src/test/java/com/javarush/vasileva/service/TestData.java new file mode 100644 index 0000000..3274469 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/service/TestData.java @@ -0,0 +1,251 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.*; +import com.javarush.vasileva.util.Value; + +import java.util.List; + +public final class TestData { + private TestData() { + } + + // QUEST + public static final Long VALID_QUEST_ID = 1L; + public static final Long NON_EXISTENT_QUEST_ID = 999L; + public static final String NON_EXISTENT_QUEST_ID_STR = "999"; + public static final String NULL_QUEST_ID_STR = null; + public static final String EMPTY_QUEST_ID_STR = ""; + public static final String QUEST_TITLE = "Test Quest"; + + // USER + public static final Long NON_EXISTENT_USER_ID = 999L; + public static final Long VALID_USER_ID = 1L; + public static final String VALID_USER_LOGIN = "testUser"; + public static final String VALID_USER_EMAIL = "test@email.com"; + public static final String VALID_USER_PASSWORD = "testPassword"; + public static final String INVALID_USER_EMAIL = "invalid@email.com"; + public static final String INVALID_USER_PASSWORD = "invalidPassword"; + public static final String EMPTY_USER_ID_STR = ""; + public static final String NULL_USER_ID_STR = null; + + // QUESTION + public static final Long VALID_QUESTION_ID = 1L; + public static final Long NON_EXISTENT_QUESTION_ID = 999L; + public static final String VALID_LABEL = "Q1"; + public static final String INVALID_LABEL = "XYZ"; + + // GAME + public static final Long VALID_GAME_ID = 100L; + public static final Long NON_EXISTENT_GAME_ID = 99999L; + + // ANSWER + public static final Long VALID_ANSWER_ID = 1L; + public static final Long NON_EXISTENT_ANSWER_ID = 999L; + public static final String VALID_ANSWER_TEXT = "Правильный ответ"; + + // STATS + public static final String WIN_LABEL = "Q1" + Value.WIN; + public static final String LOSS_LABEL = "Q2" + Value.LOSS; + + // AUTH + public static final String UNAUTHORIZED_MESSAGE = "Доступ запрещён: требуется роль администратора"; + + + /* ======================================= USER =================================== */ + public static User createValidUser() { + return User.builder() + .id(VALID_USER_ID) + .login(VALID_USER_LOGIN) + .email(VALID_USER_EMAIL) + .password(VALID_USER_PASSWORD) + .role(Role.USER) + .build(); + } + + /* ======================================= QUEST =================================== */ + + public static Quest createValidQuest() { + return Quest.builder() + .id(VALID_QUEST_ID) + .title(QUEST_TITLE) + .description("Test Quest Description") + .build(); + } + + public static Quest createQuestWithId(Long id) { + return Quest.builder() + .id(id) + .title("Quest #" + id) + .build(); + } + + public static List createMultipleQuests() { + return List.of( + createQuestWithId(1L), + createQuestWithId(2L), + createQuestWithId(3L) + ); + } + + /* ======================================= QUESTION =================================== */ + + public static Question createQuestionWithAnswers() { + Question question = Question.builder() + .generatedId(VALID_QUESTION_ID) + .label(VALID_LABEL) + .questId(VALID_QUEST_ID) + .text("Какой цвет неба?") + .build(); + List answers = List.of( + createAnswer(1L, "Синий"), + createAnswer(2L, "Голубой")); + question.setAnswers(answers); + return question; + } + + public static Question createFinalQuestion() { + return Question.builder() + .generatedId(2L) + .label("END") + .questId(VALID_QUEST_ID) + .text("Это финальный вопрос.") + .answers(null) + .build(); + } + + public static Question createSimpleQuestion() { + return Question.builder() + .generatedId(3L) + .label("Q3") + .questId(VALID_QUEST_ID) + .text("Простой вопрос.") + .build(); + } + + public static List createMultipleQuestions() { + return List.of( + createQuestionWithAnswers(), + createFinalQuestion(), + createSimpleQuestion() + ); + } + + /* ======================================= ANSWER =================================== */ + + public static Answer createAnswer(Long id, String text) { + return Answer.builder() + .id(id) + .text(text) + .build(); + } + + public static Answer createValidAnswer() { + return Answer.builder() + .id(VALID_ANSWER_ID) + .text(VALID_ANSWER_TEXT) + .build(); + } + + public static List createMultipleAnswers() { + return List.of( + createValidAnswer(), + createAnswer(2L, "Второй ответ"), + createAnswer(3L, "Третий ответ") + ); + } + + /* ======================================= GAME =================================== */ + + public static GameState createInitialGameState() { + Question currentQuestion = Question.builder() + .generatedId(VALID_QUESTION_ID) + .text("Первый вопрос квеста?") + .build(); + return GameState.builder() + .currentQuestion(currentQuestion) + .isCompleted(false) + .build(); + } + + public static GameState createNextGameState() { + Question nextQuestion = Question.builder() + .generatedId(2L) + .text("Следующий вопрос?") + .build(); + return GameState.builder() + .currentQuestion(nextQuestion) + .isCompleted(false) + .build(); + } + + public static Game createGame(Long questId, Long userId, Long currentQuestionId, GameState state) { + return Game.builder() + .questId(questId) + .userId(userId) + .currentQuestionId(currentQuestionId) + .gameState(state) + .build(); + } + + public static Game createSavedGame() { + Game game = createGame(VALID_QUEST_ID, VALID_USER_ID, 1L, createInitialGameState()); + game.setId(VALID_GAME_ID); + return game; + } + + /* ======================================= STATS =================================== */ + + public static UserStats createDefaultUserStats() { + return UserStats.builder() + .userId(VALID_USER_ID) + .total(0) + .wins(0) + .losses(0) + .build(); + } + + public static UserStats createFilledUserStats() { + return UserStats.builder() + .userId(VALID_USER_ID) + .total(5) + .wins(3) + .losses(2) + .build(); + } + + public static Question createWinQuestion() { + return Question.builder() + .label(WIN_LABEL) + .build(); + } + + public static Question createLossQuestion() { + return Question.builder() + .label(LOSS_LABEL) + .build(); + } + + /* ======================================= STATS =================================== */ + + public static User createAdminUser() { + return User.builder() + .role(Role.ADMIN) + .build(); + } + + public static User createUserUser() { + return User.builder() + .role(Role.USER) + .build(); + } + + public static User createGuestUser() { + return User.builder() + .role(Role.GUEST) + .build(); + } + + public static User createNullUser() { + return null; + } +} diff --git a/src/test/java/com/javarush/vasileva/service/UserServiceTest.java b/src/test/java/com/javarush/vasileva/service/UserServiceTest.java new file mode 100644 index 0000000..1556f13 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/service/UserServiceTest.java @@ -0,0 +1,171 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Role; +import com.javarush.vasileva.entity.User; +import com.javarush.vasileva.repository.UserRepository; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; + +import java.util.Collections; +import java.util.List; +import java.util.Optional; + +import static com.javarush.vasileva.service.TestData.*; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class UserServiceTest { + + @Mock + private UserRepository userRepository; + + @InjectMocks + private UserService userService; + + private User testUser; + + @BeforeEach + public void setUp() { + testUser = createValidUser(); + } + + @Test + @DisplayName("given user when create then repository create called") + void givenUser_whenCreate_ThenRepositoryCreateCalled() { + userService.create(testUser); + verify(userRepository, times(1)).create(testUser); + } + + @Test + @DisplayName("given user when update then repository update called") + void givenUser_whenUpdate_ThenRepositoryUpdateCalled() { + userService.update(testUser); + verify(userRepository, times(1)).update(testUser); + } + + @Test + @DisplayName("given user when delete then repository delete called") + void givenUser_whenDelete_ThenRepositoryDeleteCalled() { + userService.delete(testUser); + verify(userRepository, times(1)).delete(testUser); + } + + @Test + @DisplayName("given users in repo when get all then return list") + void givenUsersInRepo_whenGetAll_thenReturnList() { + List users = Collections.singletonList(testUser); + when(userRepository.getAll()).thenReturn(users); + + List result = userService.getAll(); + + assertEquals(1, result.size()); + assertEquals(testUser.getId(), result.get(0).getId()); + verify(userRepository, times(1)).getAll(); + } + + @Test + @DisplayName("given user id exists when find by id then return user") + void givenUserIdExists_WhenFindById_ThenReturnUser() { + long userId = testUser.getId(); + when(userRepository.findById(userId)).thenReturn(Optional.of(testUser)); + + Optional result = userService.findById(userId); + + assertTrue(result.isPresent()); + assertEquals(testUser.getId(), result.get().getId()); + verify(userRepository, times(1)).findById(userId); + } + + @Test + @DisplayName("given user id not found when find by id then return empty") + void givenUserIdNotFound_WhenFindById_ThenReturnEmpty() { + long absentUserId = NON_EXISTENT_USER_ID; + when(userRepository.findById(absentUserId)).thenReturn(Optional.empty()); + + Optional result = userService.findById(absentUserId); + + assertFalse(result.isPresent()); + verify(userRepository, times(1)).findById(absentUserId); + } + + @Test + @DisplayName("given login email password when register then repository create user with user role") + void givenLoginEmailPassword_WhenRegister_ThenRepositoryCreateUser() { + userService.register(VALID_USER_LOGIN, VALID_USER_EMAIL, VALID_USER_PASSWORD); + + User expectedUser = User.builder() + .login(VALID_USER_LOGIN) + .email(VALID_USER_EMAIL) + .password(VALID_USER_PASSWORD) + .role(Role.USER) + .build(); + + assertEquals(expectedUser.getRole(), Role.USER); + verify(userRepository, times(1)).create(expectedUser); + } + + @Test + @DisplayName("given valid email and password when login then return user") + void givenValidEmailAndPassword_WhenLogin_ThenReturnUser() { + when(userRepository.findByEmail(testUser.getEmail())).thenReturn(Optional.of(testUser)); + + Optional result = userService.login(testUser.getEmail(), testUser.getPassword()); + + assertTrue(result.isPresent()); + assertEquals(testUser.getId(), result.get().getId()); + verify(userRepository, times(1)).findByEmail(testUser.getEmail()); + } + + @Test + @DisplayName("given wrong password when login then return empty") + void givenWrongPassword_WhenLogin_ThenReturnEmpty() { + when(userRepository.findByEmail(VALID_USER_EMAIL)).thenReturn(Optional.of(testUser)); + + Optional result = userService.login(VALID_USER_EMAIL, INVALID_USER_PASSWORD); + + assertFalse(result.isPresent()); + verify(userRepository, times(1)).findByEmail(VALID_USER_EMAIL); + } + + @Test + @DisplayName("given wrong email when login then return empty") + void givenWrongLogin_WhenLogin_ThenReturnEmpty() { + when(userRepository.findByEmail(INVALID_USER_EMAIL)).thenReturn(Optional.empty()); + + Optional result = userService.login(INVALID_USER_EMAIL, VALID_USER_PASSWORD); + + assertFalse(result.isPresent()); + verify(userRepository, times(1)).findByEmail(INVALID_USER_EMAIL); + } + + @Test + @DisplayName("given valid user id string when get validated user then return user") + void givenValidUserIdString_WhenGetValidatedUser_ThenReturnUser() { + when(userRepository.findById(testUser.getId())).thenReturn(Optional.of(testUser)); + + Optional result = userService.getValidatedUser(String.valueOf(testUser.getId())); + + assertTrue(result.isPresent()); + assertEquals(testUser.getId(), result.get().getId()); + } + + @Test + @DisplayName("given empty user id string when get validated user then return empty") + void givenEmptyUserIdString_whenGetValidatedUser_thenReturnEmpty() { + Optional result = userService.getValidatedUser(EMPTY_USER_ID_STR); + assertFalse(result.isPresent()); + } + + @Test + @DisplayName("given null user when get validated user then return empty") + void givenNullUserIdString_whenGetValidatedUser_thenReturnEmpty() { + Optional result = userService.getValidatedUser(NULL_USER_ID_STR); + assertFalse(result.isPresent()); + } +} diff --git a/src/test/java/com/javarush/vasileva/service/UserStatsServiceTest.java b/src/test/java/com/javarush/vasileva/service/UserStatsServiceTest.java new file mode 100644 index 0000000..b759c16 --- /dev/null +++ b/src/test/java/com/javarush/vasileva/service/UserStatsServiceTest.java @@ -0,0 +1,102 @@ +package com.javarush.vasileva.service; + +import com.javarush.vasileva.entity.Question; +import com.javarush.vasileva.entity.UserStats; +import com.javarush.vasileva.repository.UserStatsRepository; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; + +import java.util.Optional; + +import static com.javarush.vasileva.service.TestData.*; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class UserStatsServiceTest { + + @Mock + private UserStatsRepository userStatsRepository; + + @InjectMocks + private UserStatsService userStatsService; + + @Test + @DisplayName("when createUserStats() then delegate stats creation to repository") + void testCreateUserStats() { + UserStats expectedStats = createDefaultUserStats(); + when(userStatsRepository.createUserStats(VALID_USER_ID)) + .thenReturn(expectedStats); + + UserStats result = userStatsService.createUserStats(VALID_USER_ID); + + assertEquals(expectedStats, result); + verify(userStatsRepository).createUserStats(VALID_USER_ID); + } + + @Test + @DisplayName("when getUserStats() then return user stats if it exists") + void whenGetUserStats_thenExisting() { + UserStats expectedStats = createFilledUserStats(); + when(userStatsRepository.getUserStats(VALID_USER_ID)) + .thenReturn(Optional.of(expectedStats)); + + Optional result = userStatsService.getUserStats(VALID_USER_ID); + + assertTrue(result.isPresent()); + assertEquals(expectedStats, result.get()); + verify(userStatsRepository).getUserStats(VALID_USER_ID); + } + + @Test + @DisplayName("when getUserStats() then return empty Optional if stats is not found") + void whenGetUserStats_ThenNonExisting() { + when(userStatsRepository.getUserStats(NON_EXISTENT_USER_ID)) + .thenReturn(Optional.empty()); + + Optional result = userStatsService.getUserStats(NON_EXISTENT_USER_ID); + + assertFalse(result.isPresent()); + verify(userStatsRepository).getUserStats(NON_EXISTENT_USER_ID); + } + + @Test + @DisplayName("when updateUserStats() then increase total и wins if WIN-question") + void testUpdateUserStats_WinQuestion() { + UserStats stats = createDefaultUserStats(); + Question winQuestion = createWinQuestion(); + + doAnswer(invocation -> { + invocation.getArgument(0); + return null; + }).when(userStatsRepository).updateUserStats(any(UserStats.class)); + + userStatsService.updateUserStats(winQuestion, stats); + + assertEquals(1, stats.getTotal()); + assertEquals(1, stats.getWins()); + assertEquals(0, stats.getLosses()); + verify(userStatsRepository).updateUserStats(stats); + } + + @Test + @DisplayName("when updateUserStats() then increase total и losses if LOSS-question") + void testUpdateUserStats_LossQuestion() { + UserStats stats = createDefaultUserStats(); + Question lossQuestion = createLossQuestion(); + + doAnswer(invocation -> null) + .when(userStatsRepository).updateUserStats(any(UserStats.class)); + + userStatsService.updateUserStats(lossQuestion, stats); + + assertEquals(1, stats.getTotal()); + assertEquals(0, stats.getWins()); + assertEquals(1, stats.getLosses()); + verify(userStatsRepository).updateUserStats(stats); + } +}