From 6fefe558391ffa59007d43db7ef570ef067b5e52 Mon Sep 17 00:00:00 2001 From: phc979 Date: Fri, 7 Feb 2025 17:56:31 +0900 Subject: [PATCH 01/73] init :: project init --- .../cookofking/config/SecurityConfig.java | 59 +++++++++++++++++++ .../controller/ImagemappingController.java | 10 ++++ .../cookofking/controller/LikeController.java | 10 ++++ .../cookofking/controller/PostController.java | 10 ++++ .../cookofking/controller/UserController.java | 12 ++++ .../recipe/cookofking/controller/dummy.file | 0 .../cookofking/dto/ImagemappingDto.java | 20 +++++++ .../com/recipe/cookofking/dto/LikeDto.java | 20 +++++++ .../com/recipe/cookofking/dto/PostDto.java | 24 ++++++++ .../com/recipe/cookofking/dto/UserDto.java | 24 ++++++++ .../cookofking/entity/Imagemapping.java | 29 +++++++++ .../com/recipe/cookofking/entity/Like.java | 30 ++++++++++ .../com/recipe/cookofking/entity/Post.java | 46 +++++++++++++++ .../com/recipe/cookofking/entity/User.java | 44 ++++++++++++++ .../cookofking/mapper/ImagemappingMapper.java | 33 +++++++++++ .../recipe/cookofking/mapper/LikeMapper.java | 34 +++++++++++ .../recipe/cookofking/mapper/PostMapper.java | 40 +++++++++++++ .../recipe/cookofking/mapper/UserMapper.java | 38 ++++++++++++ .../repository/ImagemappingRepository.java | 8 +++ .../cookofking/repository/LikeRepository.java | 9 +++ .../cookofking/repository/PostRepository.java | 8 +++ .../cookofking/repository/UserRepository.java | 10 ++++ .../service/ImagemappingService.java | 8 +++ .../cookofking/service/LikeService.java | 6 ++ .../cookofking/service/PostService.java | 6 ++ .../cookofking/service/UserService.java | 6 ++ 26 files changed, 544 insertions(+) create mode 100644 src/main/java/com/recipe/cookofking/config/SecurityConfig.java create mode 100644 src/main/java/com/recipe/cookofking/controller/ImagemappingController.java create mode 100644 src/main/java/com/recipe/cookofking/controller/LikeController.java create mode 100644 src/main/java/com/recipe/cookofking/controller/PostController.java create mode 100644 src/main/java/com/recipe/cookofking/controller/UserController.java create mode 100644 src/main/java/com/recipe/cookofking/controller/dummy.file create mode 100644 src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java create mode 100644 src/main/java/com/recipe/cookofking/dto/LikeDto.java create mode 100644 src/main/java/com/recipe/cookofking/dto/PostDto.java create mode 100644 src/main/java/com/recipe/cookofking/dto/UserDto.java create mode 100644 src/main/java/com/recipe/cookofking/entity/Imagemapping.java create mode 100644 src/main/java/com/recipe/cookofking/entity/Like.java create mode 100644 src/main/java/com/recipe/cookofking/entity/Post.java create mode 100644 src/main/java/com/recipe/cookofking/entity/User.java create mode 100644 src/main/java/com/recipe/cookofking/mapper/ImagemappingMapper.java create mode 100644 src/main/java/com/recipe/cookofking/mapper/LikeMapper.java create mode 100644 src/main/java/com/recipe/cookofking/mapper/PostMapper.java create mode 100644 src/main/java/com/recipe/cookofking/mapper/UserMapper.java create mode 100644 src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java create mode 100644 src/main/java/com/recipe/cookofking/repository/LikeRepository.java create mode 100644 src/main/java/com/recipe/cookofking/repository/PostRepository.java create mode 100644 src/main/java/com/recipe/cookofking/repository/UserRepository.java create mode 100644 src/main/java/com/recipe/cookofking/service/ImagemappingService.java create mode 100644 src/main/java/com/recipe/cookofking/service/LikeService.java create mode 100644 src/main/java/com/recipe/cookofking/service/PostService.java create mode 100644 src/main/java/com/recipe/cookofking/service/UserService.java diff --git a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java new file mode 100644 index 0000000..da59cd9 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java @@ -0,0 +1,59 @@ +package com.recipe.cookofking.config; + + +//@EnableMethodSecurity(prePostEnabled = true,securedEnabled = true) +//@EnableWebSecurity +//@Configuration +public class SecurityConfig { +// +// @Bean +// public SecurityFilterChain securityFilterChain(HttpSecurity security) throws Exception { +// +// security +// .csrf(csrf -> csrf.disable()); +// //.cors(cors -> cors.disable()); +// security.authorizeHttpRequests(authorizeRequests -> +// authorizeRequests +// .requestMatchers("/assets/**").permitAll() // Allow static assets +// .requestMatchers(PathRequest.toStaticResources().atCommonLocations()).permitAll() +// .requestMatchers("/u/details/**").hasRole("USER") +// .requestMatchers("/f/admin/**").hasRole("ADMIN") +// .requestMatchers("/login", "/register", "/", "/f/c", "/f", "/f/a", "/f/a/{artist_id}").permitAll() +// .anyRequest().authenticated() +// ); +// security.formLogin(form -> form +// .loginPage("/login") +// .loginProcessingUrl("/login") +// .usernameParameter("email") // 사용자 이름 필드를 이메일로 설정 +// .passwordParameter("password") // 비밀번호 필드 설정 +// .defaultSuccessUrl("/f/c")); //f/c +// +// security.logout(logout -> logout.logoutSuccessUrl("/f/c")); +// //세션 +// security.sessionManagement(httpSecuritySessionManagementConfigurer -> httpSecuritySessionManagementConfigurer +// .invalidSessionUrl("/login") +// .sessionFixation().migrateSession() //로그인후 세션 변경 +// .maximumSessions(1)//한개만 유지 +// .expiredUrl("/login"));//만료시 이동 +// +// return security.build(); +// +// } +// +// +// @Bean +// public UserDetailsService userDetailsService(UserService userService) { +// return email -> userService.searchUserByEmail(email).map(FanUserDetails::toFUDto) +// .orElseThrow(() -> new UsernameNotFoundException("User not found with email: " + email)); +// } +// +// +// +// @Bean +// public PasswordEncoder passwordEncoder() { +// +// return NoOpPasswordEncoder.getInstance(); +// return new BCryptPasswordEncoder(); +// } + +} diff --git a/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java b/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java new file mode 100644 index 0000000..74a4a15 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java @@ -0,0 +1,10 @@ +package com.recipe.cookofking.controller; + +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + +@RestController +@RequestMapping("/s3") +public class ImagemappingController { + +} diff --git a/src/main/java/com/recipe/cookofking/controller/LikeController.java b/src/main/java/com/recipe/cookofking/controller/LikeController.java new file mode 100644 index 0000000..e1844e9 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/controller/LikeController.java @@ -0,0 +1,10 @@ +package com.recipe.cookofking.controller; + +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +@RestController +@RequestMapping("/likes") + +public class LikeController { + +} diff --git a/src/main/java/com/recipe/cookofking/controller/PostController.java b/src/main/java/com/recipe/cookofking/controller/PostController.java new file mode 100644 index 0000000..4f2c295 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/controller/PostController.java @@ -0,0 +1,10 @@ +package com.recipe.cookofking.controller; + +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + +@RestController +@RequestMapping("/posts") +public class PostController { + +} diff --git a/src/main/java/com/recipe/cookofking/controller/UserController.java b/src/main/java/com/recipe/cookofking/controller/UserController.java new file mode 100644 index 0000000..68c3a38 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/controller/UserController.java @@ -0,0 +1,12 @@ +package com.recipe.cookofking.controller; + + +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + +@RestController +@RequestMapping("/users") +public class UserController { + + +} diff --git a/src/main/java/com/recipe/cookofking/controller/dummy.file b/src/main/java/com/recipe/cookofking/controller/dummy.file new file mode 100644 index 0000000..e69de29 diff --git a/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java b/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java new file mode 100644 index 0000000..8bf846f --- /dev/null +++ b/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java @@ -0,0 +1,20 @@ +package com.recipe.cookofking.dto; + +import com.recipe.cookofking.entity.Imagemapping; +import lombok.Builder; +import lombok.Value; + +import java.io.Serializable; +import java.time.Instant; + +/** + * DTO for {@link Imagemapping} + */ +@Builder +@Value +public class ImagemappingDto implements Serializable { + Integer id; + PostDto post; + String s3Url; + Instant createdDate; +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/LikeDto.java b/src/main/java/com/recipe/cookofking/dto/LikeDto.java new file mode 100644 index 0000000..4707993 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/dto/LikeDto.java @@ -0,0 +1,20 @@ +package com.recipe.cookofking.dto; + +import com.recipe.cookofking.entity.Like; +import lombok.Builder; +import lombok.Value; + +import java.io.Serializable; +import java.time.Instant; + +/** + * DTO for {@link Like} + */ +@Builder +@Value +public class LikeDto implements Serializable { + Integer id; + PostDto post; + UserDto user; + Instant createdDate; +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/PostDto.java b/src/main/java/com/recipe/cookofking/dto/PostDto.java new file mode 100644 index 0000000..9fb2e35 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/dto/PostDto.java @@ -0,0 +1,24 @@ +package com.recipe.cookofking.dto; + +import com.recipe.cookofking.entity.Post; +import lombok.Builder; +import lombok.Value; + +import java.io.Serializable; +import java.time.Instant; + +/** + * DTO for {@link Post} + */ +@Builder +@Value +public class PostDto implements Serializable { + Integer id; + UserDto user; + String title; + String content; + String ingredients; + String instructions; + Instant createdDate; + Instant modifiedDate; +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/UserDto.java b/src/main/java/com/recipe/cookofking/dto/UserDto.java new file mode 100644 index 0000000..9c5912d --- /dev/null +++ b/src/main/java/com/recipe/cookofking/dto/UserDto.java @@ -0,0 +1,24 @@ +package com.recipe.cookofking.dto; + +import com.recipe.cookofking.entity.User; +import lombok.Builder; +import lombok.Value; + +import java.io.Serializable; +import java.time.Instant; + +/** + * DTO for {@link User} + */ +@Builder +@Value +public class UserDto implements Serializable { + Integer id; + String username; + String email; + String password; + String role; + Instant createdDate; + Instant modifiedDate; + +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/entity/Imagemapping.java b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java new file mode 100644 index 0000000..a7eb0c7 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java @@ -0,0 +1,29 @@ +package com.recipe.cookofking.entity; + +import jakarta.persistence.*; +import lombok.*; + +import java.time.Instant; + +@Getter +@Builder +@NoArgsConstructor +@AllArgsConstructor +@Entity +@Table(name = "imagemapping") +public class Imagemapping { + @Id + @Column(name = "image_id", nullable = false) + private Integer id; + + @ManyToOne(fetch = FetchType.LAZY, optional = false) + @JoinColumn(name = "post_id", nullable = false) + private Post post; + + @Column(name = "s3_url", nullable = false) + private String s3Url; + + @Column(name = "createdDate") + private Instant createdDate; + +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/entity/Like.java b/src/main/java/com/recipe/cookofking/entity/Like.java new file mode 100644 index 0000000..ce9b99a --- /dev/null +++ b/src/main/java/com/recipe/cookofking/entity/Like.java @@ -0,0 +1,30 @@ +package com.recipe.cookofking.entity; + +import jakarta.persistence.*; +import lombok.*; + +import java.time.Instant; + +@Getter +@Builder +@NoArgsConstructor +@AllArgsConstructor +@Entity +@Table(name = "likes") +public class Like { + @Id + @Column(name = "like_id", nullable = false) + private Integer id; + + @ManyToOne(fetch = FetchType.LAZY, optional = false) + @JoinColumn(name = "post_id", nullable = false) + private Post post; + + @ManyToOne(fetch = FetchType.LAZY, optional = false) + @JoinColumn(name = "user_id", nullable = false) + private User user; + + @Column(name = "createdDate") + private Instant createdDate; + +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/entity/Post.java b/src/main/java/com/recipe/cookofking/entity/Post.java new file mode 100644 index 0000000..ed37bec --- /dev/null +++ b/src/main/java/com/recipe/cookofking/entity/Post.java @@ -0,0 +1,46 @@ +package com.recipe.cookofking.entity; + +import jakarta.persistence.*; +import lombok.*; +import org.hibernate.annotations.ColumnDefault; + +import java.time.Instant; + +@Getter +@Builder +@NoArgsConstructor +@AllArgsConstructor +@Entity +@Table(name = "post") +public class Post { + @Id + @Column(name = "post_id", nullable = false) + private Integer id; + + @ManyToOne(fetch = FetchType.LAZY, optional = false) + @JoinColumn(name = "user_id", nullable = false) + private User user; + + @Column(name = "title", nullable = false, length = 100) + private String title; + + @Lob + @Column(name = "content", nullable = false) + private String content; + + @Lob + @Column(name = "ingredients") + private String ingredients; + + @Lob + @Column(name = "instructions") + private String instructions; + + @Column(name = "createdDate") + private Instant createdDate; + + @ColumnDefault("CURRENT_TIMESTAMP") + @Column(name = "modifiedDate") + private Instant modifiedDate; + +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/entity/User.java b/src/main/java/com/recipe/cookofking/entity/User.java new file mode 100644 index 0000000..2967ec8 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/entity/User.java @@ -0,0 +1,44 @@ +package com.recipe.cookofking.entity; + +import jakarta.persistence.Column; +import jakarta.persistence.Entity; +import jakarta.persistence.Id; +import jakarta.persistence.Table; +import lombok.*; +import org.hibernate.annotations.ColumnDefault; + +import java.time.Instant; + + +@Getter +@Entity +@Table(name = "user") +@Builder +@NoArgsConstructor +@AllArgsConstructor +public class User { + @Id + @Column(name = "user_id", nullable = false) + private Integer id; + + @Column(name = "username", nullable = false, length = 50) + private String username; + + @Column(name = "email", nullable = false, length = 100) + private String email; + + @Column(name = "password", nullable = false) + private String password; + + @ColumnDefault("'ROLE_USER'") + @Column(name = "role", nullable = false, length = 20) + private String role; + + @Column(name = "createdDate") + private Instant createdDate; + + @ColumnDefault("CURRENT_TIMESTAMP") + @Column(name = "modifiedDate") + private Instant modifiedDate; + +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/mapper/ImagemappingMapper.java b/src/main/java/com/recipe/cookofking/mapper/ImagemappingMapper.java new file mode 100644 index 0000000..563ac67 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/mapper/ImagemappingMapper.java @@ -0,0 +1,33 @@ +package com.recipe.cookofking.mapper; + +import com.recipe.cookofking.dto.ImagemappingDto; +import com.recipe.cookofking.entity.Imagemapping; + + +public class ImagemappingMapper { + + public static ImagemappingDto toDto(Imagemapping imagemapping) { + if (imagemapping == null) { + return null; + } + return ImagemappingDto.builder() + .id(imagemapping.getId()) + .post(PostMapper.toDto(imagemapping.getPost())) + .s3Url(imagemapping.getS3Url()) + .createdDate(imagemapping.getCreatedDate()) + .build(); + } + + public static Imagemapping toEntity(ImagemappingDto imagemappingDto) { + if (imagemappingDto == null) { + return null; + } + return Imagemapping.builder() + .id(imagemappingDto.getId()) + .post(PostMapper.toEntity(imagemappingDto.getPost())) + .s3Url(imagemappingDto.getS3Url()) + .createdDate(imagemappingDto.getCreatedDate()) + .build(); + } +} + diff --git a/src/main/java/com/recipe/cookofking/mapper/LikeMapper.java b/src/main/java/com/recipe/cookofking/mapper/LikeMapper.java new file mode 100644 index 0000000..d6a64af --- /dev/null +++ b/src/main/java/com/recipe/cookofking/mapper/LikeMapper.java @@ -0,0 +1,34 @@ +package com.recipe.cookofking.mapper; + +import com.recipe.cookofking.dto.LikeDto; +import com.recipe.cookofking.entity.Like; + +public class LikeMapper { + + + public static LikeDto toDto(Like like) { + if (like == null) { + return null; + } + return LikeDto.builder() + .id(like.getId()) + .post(PostMapper.toDto(like.getPost())) + .user(UserMapper.toDto(like.getUser())) + .createdDate(like.getCreatedDate()) + .build(); + } + + public static Like toEntity(LikeDto likeDto) { + if (likeDto == null) { + return null; + } + return Like.builder() + .id(likeDto.getId()) + .post(PostMapper.toEntity(likeDto.getPost())) + .user(UserMapper.toEntity(likeDto.getUser())) + .createdDate(likeDto.getCreatedDate()) + .build(); + } + + +} diff --git a/src/main/java/com/recipe/cookofking/mapper/PostMapper.java b/src/main/java/com/recipe/cookofking/mapper/PostMapper.java new file mode 100644 index 0000000..51b19ac --- /dev/null +++ b/src/main/java/com/recipe/cookofking/mapper/PostMapper.java @@ -0,0 +1,40 @@ +package com.recipe.cookofking.mapper; + +import com.recipe.cookofking.dto.PostDto; +import com.recipe.cookofking.entity.Post; + +public class PostMapper { + public static PostDto toDto(Post post) { + if (post == null) { + return null; + } + return PostDto.builder() + .id(post.getId()) + .user(UserMapper.toDto(post.getUser())) + .title(post.getTitle()) + .content(post.getContent()) + .ingredients(post.getIngredients()) + .instructions(post.getInstructions()) + .createdDate(post.getCreatedDate()) + .modifiedDate(post.getModifiedDate()) + .build(); + } + + public static Post toEntity(PostDto postDto) { + if (postDto == null) { + return null; + } + return Post.builder() + .id(postDto.getId()) + .user(UserMapper.toEntity(postDto.getUser())) + .title(postDto.getTitle()) + .content(postDto.getContent()) + .ingredients(postDto.getIngredients()) + .instructions(postDto.getInstructions()) + .createdDate(postDto.getCreatedDate()) + .modifiedDate(postDto.getModifiedDate()) + .build(); + } + + +} diff --git a/src/main/java/com/recipe/cookofking/mapper/UserMapper.java b/src/main/java/com/recipe/cookofking/mapper/UserMapper.java new file mode 100644 index 0000000..195e652 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/mapper/UserMapper.java @@ -0,0 +1,38 @@ +package com.recipe.cookofking.mapper; + +import com.recipe.cookofking.dto.UserDto; +import com.recipe.cookofking.entity.User; + +public class UserMapper { + + public static UserDto toDto(User user) { + if (user == null) { + return null; + } + return UserDto.builder() + .id(user.getId()) + .username(user.getUsername()) + .email(user.getEmail()) + .password(user.getPassword()) + .role(user.getRole()) + .createdDate(user.getCreatedDate()) + .modifiedDate(user.getModifiedDate()) + .build(); + } + + public static User toEntity(UserDto userDto) { + if (userDto == null) { + return null; + } + return User.builder() + .id(userDto.getId()) + .username(userDto.getUsername()) + .email(userDto.getEmail()) + .password(userDto.getPassword()) + .role(userDto.getRole()) + .createdDate(userDto.getCreatedDate()) + .modifiedDate(userDto.getModifiedDate()) + .build(); + } + +} diff --git a/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java b/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java new file mode 100644 index 0000000..3285f17 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java @@ -0,0 +1,8 @@ +package com.recipe.cookofking.repository; +import com.recipe.cookofking.entity.Imagemapping; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; +@Repository +public interface ImagemappingRepository extends JpaRepository { + // Add custom query methods if needed +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/repository/LikeRepository.java b/src/main/java/com/recipe/cookofking/repository/LikeRepository.java new file mode 100644 index 0000000..32a3b9f --- /dev/null +++ b/src/main/java/com/recipe/cookofking/repository/LikeRepository.java @@ -0,0 +1,9 @@ +package com.recipe.cookofking.repository; +import com.recipe.cookofking.entity.Like; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +@Repository +public interface LikeRepository extends JpaRepository { + // Add custom query methods if needed +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/repository/PostRepository.java b/src/main/java/com/recipe/cookofking/repository/PostRepository.java new file mode 100644 index 0000000..6469422 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/repository/PostRepository.java @@ -0,0 +1,8 @@ +package com.recipe.cookofking.repository; +import com.recipe.cookofking.entity.Post; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; +@Repository +public interface PostRepository extends JpaRepository { + // Add custom query methods if needed +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/repository/UserRepository.java b/src/main/java/com/recipe/cookofking/repository/UserRepository.java new file mode 100644 index 0000000..69e8f62 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/repository/UserRepository.java @@ -0,0 +1,10 @@ +package com.recipe.cookofking.repository; + +import com.recipe.cookofking.entity.User; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +@Repository +public interface UserRepository extends JpaRepository { + // Add custom query methods if needed +} diff --git a/src/main/java/com/recipe/cookofking/service/ImagemappingService.java b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java new file mode 100644 index 0000000..9ba316b --- /dev/null +++ b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java @@ -0,0 +1,8 @@ +package com.recipe.cookofking.service; + +import org.springframework.stereotype.Service; + +@Service +public class ImagemappingService { + // Add Imagemapping related business logic here +} diff --git a/src/main/java/com/recipe/cookofking/service/LikeService.java b/src/main/java/com/recipe/cookofking/service/LikeService.java new file mode 100644 index 0000000..adb277e --- /dev/null +++ b/src/main/java/com/recipe/cookofking/service/LikeService.java @@ -0,0 +1,6 @@ +package com.recipe.cookofking.service; +import org.springframework.stereotype.Service; +@Service +public class LikeService { + // Add Like related business logic here +} diff --git a/src/main/java/com/recipe/cookofking/service/PostService.java b/src/main/java/com/recipe/cookofking/service/PostService.java new file mode 100644 index 0000000..04d78af --- /dev/null +++ b/src/main/java/com/recipe/cookofking/service/PostService.java @@ -0,0 +1,6 @@ +package com.recipe.cookofking.service; +import org.springframework.stereotype.Service; +@Service +public class PostService { + // Add Post related business logic here +} diff --git a/src/main/java/com/recipe/cookofking/service/UserService.java b/src/main/java/com/recipe/cookofking/service/UserService.java new file mode 100644 index 0000000..ffda778 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/service/UserService.java @@ -0,0 +1,6 @@ +package com.recipe.cookofking.service; +import org.springframework.stereotype.Service; +@Service +public class UserService { + // Add User related business logic here +} \ No newline at end of file From 7e09e8a775d62b96d86c51e5c13cefa85e278ea4 Mon Sep 17 00:00:00 2001 From: phc979 Date: Fri, 7 Feb 2025 18:33:46 +0900 Subject: [PATCH 02/73] fix: convert createdDate and modifiedDate from Instant to LocalDateTime --- .../recipe/cookofking/dto/ImagemappingDto.java | 3 ++- .../java/com/recipe/cookofking/dto/LikeDto.java | 3 ++- .../java/com/recipe/cookofking/dto/PostDto.java | 5 +++-- .../java/com/recipe/cookofking/dto/UserDto.java | 5 +++-- .../recipe/cookofking/entity/Imagemapping.java | 8 +++++++- .../java/com/recipe/cookofking/entity/Like.java | 13 ++++++++++--- .../java/com/recipe/cookofking/entity/Post.java | 14 ++++++++++---- .../java/com/recipe/cookofking/entity/User.java | 17 ++++++++++------- 8 files changed, 47 insertions(+), 21 deletions(-) diff --git a/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java b/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java index 8bf846f..053f111 100644 --- a/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java +++ b/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java @@ -6,6 +6,7 @@ import java.io.Serializable; import java.time.Instant; +import java.time.LocalDateTime; /** * DTO for {@link Imagemapping} @@ -16,5 +17,5 @@ public class ImagemappingDto implements Serializable { Integer id; PostDto post; String s3Url; - Instant createdDate; + LocalDateTime createdDate; } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/LikeDto.java b/src/main/java/com/recipe/cookofking/dto/LikeDto.java index 4707993..b5cbc97 100644 --- a/src/main/java/com/recipe/cookofking/dto/LikeDto.java +++ b/src/main/java/com/recipe/cookofking/dto/LikeDto.java @@ -6,6 +6,7 @@ import java.io.Serializable; import java.time.Instant; +import java.time.LocalDateTime; /** * DTO for {@link Like} @@ -16,5 +17,5 @@ public class LikeDto implements Serializable { Integer id; PostDto post; UserDto user; - Instant createdDate; + LocalDateTime createdDate; } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/PostDto.java b/src/main/java/com/recipe/cookofking/dto/PostDto.java index 9fb2e35..edcc2ab 100644 --- a/src/main/java/com/recipe/cookofking/dto/PostDto.java +++ b/src/main/java/com/recipe/cookofking/dto/PostDto.java @@ -6,6 +6,7 @@ import java.io.Serializable; import java.time.Instant; +import java.time.LocalDateTime; /** * DTO for {@link Post} @@ -19,6 +20,6 @@ public class PostDto implements Serializable { String content; String ingredients; String instructions; - Instant createdDate; - Instant modifiedDate; + LocalDateTime createdDate; + LocalDateTime modifiedDate; } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/UserDto.java b/src/main/java/com/recipe/cookofking/dto/UserDto.java index 9c5912d..70c9fed 100644 --- a/src/main/java/com/recipe/cookofking/dto/UserDto.java +++ b/src/main/java/com/recipe/cookofking/dto/UserDto.java @@ -6,6 +6,7 @@ import java.io.Serializable; import java.time.Instant; +import java.time.LocalDateTime; /** * DTO for {@link User} @@ -18,7 +19,7 @@ public class UserDto implements Serializable { String email; String password; String role; - Instant createdDate; - Instant modifiedDate; + LocalDateTime createdDate; + LocalDateTime modifiedDate; } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/entity/Imagemapping.java b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java index a7eb0c7..0f2bb91 100644 --- a/src/main/java/com/recipe/cookofking/entity/Imagemapping.java +++ b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java @@ -2,8 +2,12 @@ import jakarta.persistence.*; import lombok.*; +import org.springframework.data.annotation.CreatedDate; +import org.springframework.data.annotation.LastModifiedDate; +import java.sql.Timestamp; import java.time.Instant; +import java.time.LocalDateTime; @Getter @Builder @@ -14,6 +18,7 @@ public class Imagemapping { @Id @Column(name = "image_id", nullable = false) + @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer id; @ManyToOne(fetch = FetchType.LAZY, optional = false) @@ -24,6 +29,7 @@ public class Imagemapping { private String s3Url; @Column(name = "createdDate") - private Instant createdDate; + @CreatedDate + private LocalDateTime createdDate; } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/entity/Like.java b/src/main/java/com/recipe/cookofking/entity/Like.java index ce9b99a..f5cb9eb 100644 --- a/src/main/java/com/recipe/cookofking/entity/Like.java +++ b/src/main/java/com/recipe/cookofking/entity/Like.java @@ -2,8 +2,12 @@ import jakarta.persistence.*; import lombok.*; +import org.springframework.data.annotation.CreatedDate; +import org.springframework.data.annotation.LastModifiedDate; +import java.sql.Timestamp; import java.time.Instant; +import java.time.LocalDateTime; @Getter @Builder @@ -14,17 +18,20 @@ public class Like { @Id @Column(name = "like_id", nullable = false) + @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer id; - @ManyToOne(fetch = FetchType.LAZY, optional = false) + @ManyToOne(fetch = FetchType.LAZY) @JoinColumn(name = "post_id", nullable = false) private Post post; - @ManyToOne(fetch = FetchType.LAZY, optional = false) + @ManyToOne(fetch = FetchType.LAZY) @JoinColumn(name = "user_id", nullable = false) private User user; @Column(name = "createdDate") - private Instant createdDate; + @CreatedDate + private LocalDateTime createdDate; + } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/entity/Post.java b/src/main/java/com/recipe/cookofking/entity/Post.java index ed37bec..f6baa8a 100644 --- a/src/main/java/com/recipe/cookofking/entity/Post.java +++ b/src/main/java/com/recipe/cookofking/entity/Post.java @@ -3,8 +3,12 @@ import jakarta.persistence.*; import lombok.*; import org.hibernate.annotations.ColumnDefault; +import org.springframework.data.annotation.CreatedDate; +import org.springframework.data.annotation.LastModifiedDate; +import java.sql.Timestamp; import java.time.Instant; +import java.time.LocalDateTime; @Getter @Builder @@ -15,9 +19,10 @@ public class Post { @Id @Column(name = "post_id", nullable = false) + @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer id; - @ManyToOne(fetch = FetchType.LAZY, optional = false) + @ManyToOne(fetch = FetchType.LAZY) @JoinColumn(name = "user_id", nullable = false) private User user; @@ -37,10 +42,11 @@ public class Post { private String instructions; @Column(name = "createdDate") - private Instant createdDate; + @CreatedDate + private LocalDateTime createdDate; - @ColumnDefault("CURRENT_TIMESTAMP") @Column(name = "modifiedDate") - private Instant modifiedDate; + @LastModifiedDate + private LocalDateTime modifiedDate; } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/entity/User.java b/src/main/java/com/recipe/cookofking/entity/User.java index 2967ec8..7935079 100644 --- a/src/main/java/com/recipe/cookofking/entity/User.java +++ b/src/main/java/com/recipe/cookofking/entity/User.java @@ -1,13 +1,14 @@ package com.recipe.cookofking.entity; -import jakarta.persistence.Column; -import jakarta.persistence.Entity; -import jakarta.persistence.Id; -import jakarta.persistence.Table; +import jakarta.persistence.*; import lombok.*; import org.hibernate.annotations.ColumnDefault; +import org.springframework.data.annotation.CreatedDate; +import org.springframework.data.annotation.LastModifiedDate; +import java.sql.Timestamp; import java.time.Instant; +import java.time.LocalDateTime; @Getter @@ -19,6 +20,7 @@ public class User { @Id @Column(name = "user_id", nullable = false) + @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer id; @Column(name = "username", nullable = false, length = 50) @@ -35,10 +37,11 @@ public class User { private String role; @Column(name = "createdDate") - private Instant createdDate; + @CreatedDate + private LocalDateTime createdDate; - @ColumnDefault("CURRENT_TIMESTAMP") @Column(name = "modifiedDate") - private Instant modifiedDate; + @LastModifiedDate + private LocalDateTime modifiedDate; } \ No newline at end of file From e444e35b35d6b8d42467b4f775578ea65abe5dc6 Mon Sep 17 00:00:00 2001 From: phc979 Date: Fri, 7 Feb 2025 18:34:48 +0900 Subject: [PATCH 03/73] fix: convert createdDate and modifiedDate from Instant to LocalDateTime --- src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java | 1 - src/main/java/com/recipe/cookofking/dto/LikeDto.java | 1 - src/main/java/com/recipe/cookofking/dto/PostDto.java | 1 - src/main/java/com/recipe/cookofking/dto/UserDto.java | 1 - src/main/java/com/recipe/cookofking/entity/Imagemapping.java | 4 ---- src/main/java/com/recipe/cookofking/entity/Like.java | 4 ---- src/main/java/com/recipe/cookofking/entity/Post.java | 4 +--- src/main/java/com/recipe/cookofking/entity/User.java | 2 -- 8 files changed, 1 insertion(+), 17 deletions(-) diff --git a/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java b/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java index 053f111..942107e 100644 --- a/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java +++ b/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java @@ -5,7 +5,6 @@ import lombok.Value; import java.io.Serializable; -import java.time.Instant; import java.time.LocalDateTime; /** diff --git a/src/main/java/com/recipe/cookofking/dto/LikeDto.java b/src/main/java/com/recipe/cookofking/dto/LikeDto.java index b5cbc97..f49f0b9 100644 --- a/src/main/java/com/recipe/cookofking/dto/LikeDto.java +++ b/src/main/java/com/recipe/cookofking/dto/LikeDto.java @@ -5,7 +5,6 @@ import lombok.Value; import java.io.Serializable; -import java.time.Instant; import java.time.LocalDateTime; /** diff --git a/src/main/java/com/recipe/cookofking/dto/PostDto.java b/src/main/java/com/recipe/cookofking/dto/PostDto.java index edcc2ab..afffb6e 100644 --- a/src/main/java/com/recipe/cookofking/dto/PostDto.java +++ b/src/main/java/com/recipe/cookofking/dto/PostDto.java @@ -5,7 +5,6 @@ import lombok.Value; import java.io.Serializable; -import java.time.Instant; import java.time.LocalDateTime; /** diff --git a/src/main/java/com/recipe/cookofking/dto/UserDto.java b/src/main/java/com/recipe/cookofking/dto/UserDto.java index 70c9fed..88c0e85 100644 --- a/src/main/java/com/recipe/cookofking/dto/UserDto.java +++ b/src/main/java/com/recipe/cookofking/dto/UserDto.java @@ -5,7 +5,6 @@ import lombok.Value; import java.io.Serializable; -import java.time.Instant; import java.time.LocalDateTime; /** diff --git a/src/main/java/com/recipe/cookofking/entity/Imagemapping.java b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java index 0f2bb91..0a66766 100644 --- a/src/main/java/com/recipe/cookofking/entity/Imagemapping.java +++ b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java @@ -3,10 +3,6 @@ import jakarta.persistence.*; import lombok.*; import org.springframework.data.annotation.CreatedDate; -import org.springframework.data.annotation.LastModifiedDate; - -import java.sql.Timestamp; -import java.time.Instant; import java.time.LocalDateTime; @Getter diff --git a/src/main/java/com/recipe/cookofking/entity/Like.java b/src/main/java/com/recipe/cookofking/entity/Like.java index f5cb9eb..03aea63 100644 --- a/src/main/java/com/recipe/cookofking/entity/Like.java +++ b/src/main/java/com/recipe/cookofking/entity/Like.java @@ -3,10 +3,6 @@ import jakarta.persistence.*; import lombok.*; import org.springframework.data.annotation.CreatedDate; -import org.springframework.data.annotation.LastModifiedDate; - -import java.sql.Timestamp; -import java.time.Instant; import java.time.LocalDateTime; @Getter diff --git a/src/main/java/com/recipe/cookofking/entity/Post.java b/src/main/java/com/recipe/cookofking/entity/Post.java index f6baa8a..ad048ff 100644 --- a/src/main/java/com/recipe/cookofking/entity/Post.java +++ b/src/main/java/com/recipe/cookofking/entity/Post.java @@ -2,12 +2,10 @@ import jakarta.persistence.*; import lombok.*; -import org.hibernate.annotations.ColumnDefault; import org.springframework.data.annotation.CreatedDate; import org.springframework.data.annotation.LastModifiedDate; -import java.sql.Timestamp; -import java.time.Instant; + import java.time.LocalDateTime; @Getter diff --git a/src/main/java/com/recipe/cookofking/entity/User.java b/src/main/java/com/recipe/cookofking/entity/User.java index 7935079..835e822 100644 --- a/src/main/java/com/recipe/cookofking/entity/User.java +++ b/src/main/java/com/recipe/cookofking/entity/User.java @@ -6,8 +6,6 @@ import org.springframework.data.annotation.CreatedDate; import org.springframework.data.annotation.LastModifiedDate; -import java.sql.Timestamp; -import java.time.Instant; import java.time.LocalDateTime; From b8b62cbc5eef822dad2342339b7076943fc89270 Mon Sep 17 00:00:00 2001 From: phc979 Date: Sun, 9 Feb 2025 15:04:14 +0900 Subject: [PATCH 04/73] feature-front-backend-post-form --- build.gradle | 1 + .../Cld2CookofKingRecipeApplication.java | 2 + .../recipe/cookofking/config/AwsS3Config.java | 33 ++ .../cookofking/config/SecurityConfig.java | 22 +- .../controller/ImagemappingController.java | 24 +- .../cookofking/controller/PostController.java | 9 +- .../controller/PostRestController.java | 28 ++ .../com/recipe/cookofking/dto/LikeDto.java | 1 + .../dto/image/ImageValidationDto.java | 18 + .../dto/{ => image}/ImagemappingDto.java | 4 +- .../cookofking/dto/{ => post}/PostDto.java | 4 +- .../dto/post/RecipeSubmissionDto.java | 10 + .../cookofking/entity/Imagemapping.java | 21 +- .../com/recipe/cookofking/entity/Like.java | 3 + .../com/recipe/cookofking/entity/Post.java | 12 +- .../com/recipe/cookofking/entity/User.java | 2 + .../cookofking/mapper/ImagemappingMapper.java | 6 +- .../recipe/cookofking/mapper/PostMapper.java | 2 +- .../repository/ImagemappingRepository.java | 5 +- .../service/ImagemappingService.java | 70 +++- .../cookofking/service/PostService.java | 24 +- src/main/resources/static/bootstrap.min.css | 7 + src/main/resources/templates/Post/post.html | 324 ++++++++++++++++++ 23 files changed, 611 insertions(+), 21 deletions(-) create mode 100644 src/main/java/com/recipe/cookofking/config/AwsS3Config.java create mode 100644 src/main/java/com/recipe/cookofking/controller/PostRestController.java create mode 100644 src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java rename src/main/java/com/recipe/cookofking/dto/{ => image}/ImagemappingDto.java (76%) rename src/main/java/com/recipe/cookofking/dto/{ => post}/PostDto.java (79%) create mode 100644 src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java create mode 100644 src/main/resources/static/bootstrap.min.css create mode 100644 src/main/resources/templates/Post/post.html diff --git a/build.gradle b/build.gradle index ab01e42..597ca3d 100644 --- a/build.gradle +++ b/build.gradle @@ -31,6 +31,7 @@ dependencies { implementation 'org.springframework.boot:spring-boot-starter-web' implementation 'org.springframework.boot:spring-boot-starter-websocket' implementation 'org.thymeleaf.extras:thymeleaf-extras-springsecurity6' + implementation 'software.amazon.awssdk:s3:2.20.147' compileOnly 'org.projectlombok:lombok' developmentOnly 'org.springframework.boot:spring-boot-devtools' runtimeOnly 'com.mysql:mysql-connector-j' diff --git a/src/main/java/com/recipe/cookofking/Cld2CookofKingRecipeApplication.java b/src/main/java/com/recipe/cookofking/Cld2CookofKingRecipeApplication.java index 9ebc771..43766e9 100644 --- a/src/main/java/com/recipe/cookofking/Cld2CookofKingRecipeApplication.java +++ b/src/main/java/com/recipe/cookofking/Cld2CookofKingRecipeApplication.java @@ -2,8 +2,10 @@ import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; +import org.springframework.data.jpa.repository.config.EnableJpaAuditing; @SpringBootApplication +@EnableJpaAuditing public class Cld2CookofKingRecipeApplication { public static void main(String[] args) { diff --git a/src/main/java/com/recipe/cookofking/config/AwsS3Config.java b/src/main/java/com/recipe/cookofking/config/AwsS3Config.java new file mode 100644 index 0000000..36505f7 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/config/AwsS3Config.java @@ -0,0 +1,33 @@ +package com.recipe.cookofking.config; + + +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import software.amazon.awssdk.auth.credentials.AwsBasicCredentials; +import software.amazon.awssdk.auth.credentials.StaticCredentialsProvider; +import software.amazon.awssdk.regions.Region; +import software.amazon.awssdk.services.s3.S3Client; + +@Configuration +public class AwsS3Config { + + @Value("${cloud.aws.credentials.access-key}") + private String accessKey; + + @Value("${cloud.aws.credentials.secret-key}") + private String secretKey; + + @Value("${cloud.aws.region.static}") + private String region; + + + @Bean + public S3Client s3Client() { + return S3Client.builder() + .region(Region.of(region)) // 주입된 리전 값 사용 + .credentialsProvider(StaticCredentialsProvider.create( + AwsBasicCredentials.create(accessKey, secretKey))) + .build(); + } +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java index da59cd9..5a61be1 100644 --- a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java +++ b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java @@ -4,7 +4,27 @@ //@EnableMethodSecurity(prePostEnabled = true,securedEnabled = true) //@EnableWebSecurity //@Configuration + +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.security.config.annotation.web.builders.HttpSecurity; +import org.springframework.security.web.SecurityFilterChain; + +@Configuration public class SecurityConfig { + + @Bean + public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception { + http + .authorizeHttpRequests(auth -> auth + .anyRequest().permitAll() // 모든 요청 허용 + ) + .csrf(csrf -> csrf.disable()) // CSRF 비활성화 (개발 환경) + .formLogin(form -> form.disable()); // 폼 로그인 비활성화 + + return http.build(); + } +} // // @Bean // public SecurityFilterChain securityFilterChain(HttpSecurity security) throws Exception { @@ -56,4 +76,4 @@ public class SecurityConfig { // return new BCryptPasswordEncoder(); // } -} +//} diff --git a/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java b/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java index 74a4a15..e3c2605 100644 --- a/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java +++ b/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java @@ -1,10 +1,30 @@ package com.recipe.cookofking.controller; +import com.recipe.cookofking.dto.image.ImagemappingDto; +import com.recipe.cookofking.service.ImagemappingService; +import lombok.RequiredArgsConstructor; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestParam; import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.multipart.MultipartFile; @RestController -@RequestMapping("/s3") +@RequestMapping("/upload-image") +@RequiredArgsConstructor public class ImagemappingController { -} + private final ImagemappingService imagemappingService; + + @PostMapping + public ResponseEntity uploadImage(@RequestParam("file") MultipartFile file) { + try { + ImagemappingDto uploadedImage = imagemappingService.uploadImage(file); + return ResponseEntity.ok(uploadedImage); + } catch (Exception e) { + // 에러 메시지와 스택 트레이스를 함께 반환 + return ResponseEntity.status(500).body("Error: " + e.getMessage()); + } + } +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/controller/PostController.java b/src/main/java/com/recipe/cookofking/controller/PostController.java index 4f2c295..e239f71 100644 --- a/src/main/java/com/recipe/cookofking/controller/PostController.java +++ b/src/main/java/com/recipe/cookofking/controller/PostController.java @@ -1,10 +1,13 @@ package com.recipe.cookofking.controller; +import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RestController; -@RestController -@RequestMapping("/posts") +@Controller public class PostController { + @RequestMapping("/") + public String showForm() { + return "Post/post"; // 레시피 작성 폼 렌더링 + } } diff --git a/src/main/java/com/recipe/cookofking/controller/PostRestController.java b/src/main/java/com/recipe/cookofking/controller/PostRestController.java new file mode 100644 index 0000000..2e1266a --- /dev/null +++ b/src/main/java/com/recipe/cookofking/controller/PostRestController.java @@ -0,0 +1,28 @@ +package com.recipe.cookofking.controller; + +import com.recipe.cookofking.dto.post.RecipeSubmissionDto; +import com.recipe.cookofking.service.ImagemappingService; +import com.recipe.cookofking.service.PostService; +import lombok.RequiredArgsConstructor; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.*; + +@RestController +@RequestMapping("/api") +@RequiredArgsConstructor +public class PostRestController { + + private final PostService postService; + private final ImagemappingService imagemappingService; + // 레시피 저장 API + @PostMapping("/submit-recipe") + public ResponseEntity submitRecipe(@RequestBody RecipeSubmissionDto submissionDto) { + // 1. 이미지 검증 + imagemappingService.validateAndMarkPermanent(submissionDto.getValidationData()); + + // 2. 레시피 저장 + postService.savePost(submissionDto.getRecipeData()); + + return ResponseEntity.ok("레시피가 성공적으로 저장되었습니다!"); + } +} diff --git a/src/main/java/com/recipe/cookofking/dto/LikeDto.java b/src/main/java/com/recipe/cookofking/dto/LikeDto.java index f49f0b9..871c24a 100644 --- a/src/main/java/com/recipe/cookofking/dto/LikeDto.java +++ b/src/main/java/com/recipe/cookofking/dto/LikeDto.java @@ -1,5 +1,6 @@ package com.recipe.cookofking.dto; +import com.recipe.cookofking.dto.post.PostDto; import com.recipe.cookofking.entity.Like; import lombok.Builder; import lombok.Value; diff --git a/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java b/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java new file mode 100644 index 0000000..3e25b31 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java @@ -0,0 +1,18 @@ +package com.recipe.cookofking.dto.image; + +import lombok.Data; + +import java.util.List; + +@Data +public class ImageValidationDto { + private Integer mainImageId; + private String mainImageUrl; + private List stepImages; + + @Data + public static class StepImageDto { + private Integer imageId; + private String imageUrl; + } +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java b/src/main/java/com/recipe/cookofking/dto/image/ImagemappingDto.java similarity index 76% rename from src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java rename to src/main/java/com/recipe/cookofking/dto/image/ImagemappingDto.java index 942107e..ac9296b 100644 --- a/src/main/java/com/recipe/cookofking/dto/ImagemappingDto.java +++ b/src/main/java/com/recipe/cookofking/dto/image/ImagemappingDto.java @@ -1,5 +1,6 @@ -package com.recipe.cookofking.dto; +package com.recipe.cookofking.dto.image; +import com.recipe.cookofking.dto.post.PostDto; import com.recipe.cookofking.entity.Imagemapping; import lombok.Builder; import lombok.Value; @@ -16,5 +17,6 @@ public class ImagemappingDto implements Serializable { Integer id; PostDto post; String s3Url; + boolean isTemp; LocalDateTime createdDate; } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/PostDto.java b/src/main/java/com/recipe/cookofking/dto/post/PostDto.java similarity index 79% rename from src/main/java/com/recipe/cookofking/dto/PostDto.java rename to src/main/java/com/recipe/cookofking/dto/post/PostDto.java index afffb6e..fc8d9df 100644 --- a/src/main/java/com/recipe/cookofking/dto/PostDto.java +++ b/src/main/java/com/recipe/cookofking/dto/post/PostDto.java @@ -1,5 +1,6 @@ -package com.recipe.cookofking.dto; +package com.recipe.cookofking.dto.post; +import com.recipe.cookofking.dto.UserDto; import com.recipe.cookofking.entity.Post; import lombok.Builder; import lombok.Value; @@ -19,6 +20,7 @@ public class PostDto implements Serializable { String content; String ingredients; String instructions; + String mainImageS3URL; LocalDateTime createdDate; LocalDateTime modifiedDate; } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java b/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java new file mode 100644 index 0000000..fdab3f9 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java @@ -0,0 +1,10 @@ +package com.recipe.cookofking.dto.post; + +import com.recipe.cookofking.dto.image.ImageValidationDto; +import lombok.Data; + +@Data +public class RecipeSubmissionDto { + private ImageValidationDto validationData; + private PostDto recipeData; +} diff --git a/src/main/java/com/recipe/cookofking/entity/Imagemapping.java b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java index 0a66766..b2ab25d 100644 --- a/src/main/java/com/recipe/cookofking/entity/Imagemapping.java +++ b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java @@ -3,6 +3,8 @@ import jakarta.persistence.*; import lombok.*; import org.springframework.data.annotation.CreatedDate; +import org.springframework.data.jpa.domain.support.AuditingEntityListener; + import java.time.LocalDateTime; @Getter @@ -11,21 +13,30 @@ @AllArgsConstructor @Entity @Table(name = "imagemapping") +@EntityListeners(AuditingEntityListener.class) // 자동으로 생성일자 저장 public class Imagemapping { + @Id - @Column(name = "image_id", nullable = false) @GeneratedValue(strategy = GenerationType.IDENTITY) + @Column(name = "image_id", nullable = false) private Integer id; - @ManyToOne(fetch = FetchType.LAZY, optional = false) - @JoinColumn(name = "post_id", nullable = false) + @ManyToOne(fetch = FetchType.LAZY, optional = true) + @JoinColumn(name = "post_id", nullable = true) private Post post; @Column(name = "s3_url", nullable = false) private String s3Url; - @Column(name = "createdDate") + @Column(name = "is_temp", nullable = false) + private boolean isTemp; // 이미지 임시 상태 플래그 (작성 중 true, 작성 완료 false) + @CreatedDate + @Column(name = "created_date", updatable = false) private LocalDateTime createdDate; -} \ No newline at end of file + // isTemp 상태를 변경하는 명확한 메서드 + public void markAsPermanent() { + this.isTemp = false; + } +} diff --git a/src/main/java/com/recipe/cookofking/entity/Like.java b/src/main/java/com/recipe/cookofking/entity/Like.java index 03aea63..0c23e89 100644 --- a/src/main/java/com/recipe/cookofking/entity/Like.java +++ b/src/main/java/com/recipe/cookofking/entity/Like.java @@ -3,6 +3,8 @@ import jakarta.persistence.*; import lombok.*; import org.springframework.data.annotation.CreatedDate; +import org.springframework.data.jpa.domain.support.AuditingEntityListener; + import java.time.LocalDateTime; @Getter @@ -11,6 +13,7 @@ @AllArgsConstructor @Entity @Table(name = "likes") +@EntityListeners(AuditingEntityListener.class) // 자동으로 생성일자 저장 public class Like { @Id @Column(name = "like_id", nullable = false) diff --git a/src/main/java/com/recipe/cookofking/entity/Post.java b/src/main/java/com/recipe/cookofking/entity/Post.java index ad048ff..044c240 100644 --- a/src/main/java/com/recipe/cookofking/entity/Post.java +++ b/src/main/java/com/recipe/cookofking/entity/Post.java @@ -4,6 +4,7 @@ import lombok.*; import org.springframework.data.annotation.CreatedDate; import org.springframework.data.annotation.LastModifiedDate; +import org.springframework.data.jpa.domain.support.AuditingEntityListener; import java.time.LocalDateTime; @@ -14,14 +15,15 @@ @AllArgsConstructor @Entity @Table(name = "post") +@EntityListeners(AuditingEntityListener.class) // 자동으로 생성일자 저장 public class Post { @Id @Column(name = "post_id", nullable = false) @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer id; - @ManyToOne(fetch = FetchType.LAZY) - @JoinColumn(name = "user_id", nullable = false) + @ManyToOne(fetch = FetchType.LAZY, optional = true) + @JoinColumn(name = "user_id", nullable = true) private User user; @Column(name = "title", nullable = false, length = 100) @@ -36,9 +38,13 @@ public class Post { private String ingredients; @Lob - @Column(name = "instructions") + @Column(name = "instructions", columnDefinition = "TEXT") private String instructions; + @Lob + @Column(name = "mainImageS3URL") + private String mainImageS3URL; + @Column(name = "createdDate") @CreatedDate private LocalDateTime createdDate; diff --git a/src/main/java/com/recipe/cookofking/entity/User.java b/src/main/java/com/recipe/cookofking/entity/User.java index 835e822..57bf878 100644 --- a/src/main/java/com/recipe/cookofking/entity/User.java +++ b/src/main/java/com/recipe/cookofking/entity/User.java @@ -5,6 +5,7 @@ import org.hibernate.annotations.ColumnDefault; import org.springframework.data.annotation.CreatedDate; import org.springframework.data.annotation.LastModifiedDate; +import org.springframework.data.jpa.domain.support.AuditingEntityListener; import java.time.LocalDateTime; @@ -15,6 +16,7 @@ @Builder @NoArgsConstructor @AllArgsConstructor +@EntityListeners(AuditingEntityListener.class) // 자동으로 생성일자 저장 public class User { @Id @Column(name = "user_id", nullable = false) diff --git a/src/main/java/com/recipe/cookofking/mapper/ImagemappingMapper.java b/src/main/java/com/recipe/cookofking/mapper/ImagemappingMapper.java index 563ac67..ae6136f 100644 --- a/src/main/java/com/recipe/cookofking/mapper/ImagemappingMapper.java +++ b/src/main/java/com/recipe/cookofking/mapper/ImagemappingMapper.java @@ -1,6 +1,6 @@ package com.recipe.cookofking.mapper; -import com.recipe.cookofking.dto.ImagemappingDto; +import com.recipe.cookofking.dto.image.ImagemappingDto; import com.recipe.cookofking.entity.Imagemapping; @@ -14,6 +14,7 @@ public static ImagemappingDto toDto(Imagemapping imagemapping) { .id(imagemapping.getId()) .post(PostMapper.toDto(imagemapping.getPost())) .s3Url(imagemapping.getS3Url()) + .isTemp(imagemapping.isTemp()) .createdDate(imagemapping.getCreatedDate()) .build(); } @@ -26,8 +27,11 @@ public static Imagemapping toEntity(ImagemappingDto imagemappingDto) { .id(imagemappingDto.getId()) .post(PostMapper.toEntity(imagemappingDto.getPost())) .s3Url(imagemappingDto.getS3Url()) + .isTemp(imagemappingDto.isTemp()) .createdDate(imagemappingDto.getCreatedDate()) .build(); } + + } diff --git a/src/main/java/com/recipe/cookofking/mapper/PostMapper.java b/src/main/java/com/recipe/cookofking/mapper/PostMapper.java index 51b19ac..c6d4bc8 100644 --- a/src/main/java/com/recipe/cookofking/mapper/PostMapper.java +++ b/src/main/java/com/recipe/cookofking/mapper/PostMapper.java @@ -1,6 +1,6 @@ package com.recipe.cookofking.mapper; -import com.recipe.cookofking.dto.PostDto; +import com.recipe.cookofking.dto.post.PostDto; import com.recipe.cookofking.entity.Post; public class PostMapper { diff --git a/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java b/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java index 3285f17..89e38f7 100644 --- a/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java +++ b/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java @@ -2,7 +2,10 @@ import com.recipe.cookofking.entity.Imagemapping; import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.stereotype.Repository; + +import java.util.Optional; + @Repository public interface ImagemappingRepository extends JpaRepository { - // Add custom query methods if needed + Optional findByIdAndS3Url(Integer id, String s3Url); } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/service/ImagemappingService.java b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java index 9ba316b..b4163bf 100644 --- a/src/main/java/com/recipe/cookofking/service/ImagemappingService.java +++ b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java @@ -1,8 +1,76 @@ package com.recipe.cookofking.service; +import com.recipe.cookofking.dto.image.ImageValidationDto; +import com.recipe.cookofking.dto.image.ImagemappingDto; +import com.recipe.cookofking.entity.Imagemapping; +import com.recipe.cookofking.mapper.ImagemappingMapper; +import com.recipe.cookofking.repository.ImagemappingRepository; +import lombok.RequiredArgsConstructor; +import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Service; +import org.springframework.web.multipart.MultipartFile; +import software.amazon.awssdk.services.s3.S3Client; +import software.amazon.awssdk.services.s3.model.PutObjectRequest; + +import java.io.IOException; +import java.util.UUID; @Service +@RequiredArgsConstructor public class ImagemappingService { - // Add Imagemapping related business logic here + + private final S3Client s3Client; + private final ImagemappingRepository imagemappingRepository; + + // application.properties에서 버킷 이름 주입 + @Value("${cloud.aws.s3.bucket}") + private String bucketName; + + public ImagemappingDto uploadImage(MultipartFile file) throws IOException { + // S3에 업로드할 파일명 생성 + String fileName = "recipes/" + UUID.randomUUID() + "_" + file.getOriginalFilename(); + + // S3 업로드 요청 생성 + PutObjectRequest putObjectRequest = PutObjectRequest.builder() + .bucket(bucketName) // 주입된 버킷 이름 사용 + .key(fileName) + .build(); + + // S3에 파일 업로드 + s3Client.putObject(putObjectRequest, software.amazon.awssdk.core.sync.RequestBody.fromInputStream(file.getInputStream(), file.getSize())); + + // 업로드된 파일의 S3 URL 생성 + String s3Url = "https://" + bucketName + ".s3.amazonaws.com/" + fileName; + + // DB에 이미지 정보 저장 + Imagemapping imagemapping = Imagemapping.builder() + .s3Url(s3Url) + .isTemp(true) + .build(); + + Imagemapping savedImage = imagemappingRepository.save(imagemapping); + + // 저장된 이미지 정보를 DTO로 변환하여 반환 + return ImagemappingMapper.toDto(savedImage); + } + + public void validateAndMarkPermanent(ImageValidationDto validationDto) { + // 메인 이미지 검증 + imagemappingRepository.findByIdAndS3Url(validationDto.getMainImageId(), validationDto.getMainImageUrl()) + .ifPresent(image -> { + image.markAsPermanent(); + imagemappingRepository.save(image); + }); + + // 조리 순서 이미지 검증 + validationDto.getStepImages().forEach(stepImage -> { + imagemappingRepository.findByIdAndS3Url(stepImage.getImageId(), stepImage.getImageUrl()) + .ifPresent(image -> { + image.markAsPermanent(); + imagemappingRepository.save(image); + }); + }); + } + + } diff --git a/src/main/java/com/recipe/cookofking/service/PostService.java b/src/main/java/com/recipe/cookofking/service/PostService.java index 04d78af..8820ae5 100644 --- a/src/main/java/com/recipe/cookofking/service/PostService.java +++ b/src/main/java/com/recipe/cookofking/service/PostService.java @@ -1,6 +1,28 @@ package com.recipe.cookofking.service; +import com.recipe.cookofking.dto.post.PostDto; +import com.recipe.cookofking.entity.Post; +import com.recipe.cookofking.repository.PostRepository; +import lombok.RequiredArgsConstructor; import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + @Service +@RequiredArgsConstructor public class PostService { - // Add Post related business logic here + private final PostRepository postRepository; + + @Transactional + public void savePost(PostDto postDto) { + // 1. Post 엔티티 생성 + Post post = Post.builder() + .title(postDto.getTitle()) + .content(postDto.getContent()) + .ingredients(postDto.getIngredients()) // JSON 문자열로 저장 + .instructions(postDto.getInstructions()) // JSON 문자열로 저장 + .mainImageS3URL(postDto.getMainImageS3URL()) + .build(); + + // 2. 레시피 저장 + Post savedPost = postRepository.save(post); + } } diff --git a/src/main/resources/static/bootstrap.min.css b/src/main/resources/static/bootstrap.min.css new file mode 100644 index 0000000..edfbbb0 --- /dev/null +++ b/src/main/resources/static/bootstrap.min.css @@ -0,0 +1,7 @@ +@charset "UTF-8";/*! + * Bootstrap v5.0.2 (https://getbootstrap.com/) + * Copyright 2011-2021 The Bootstrap Authors + * Copyright 2011-2021 Twitter, Inc. + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + */:root{--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-white:#fff;--bs-gray:#6c757d;--bs-gray-dark:#343a40;--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-font-sans-serif:system-ui,-apple-system,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans","Liberation Sans",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))}*,::after,::before{box-sizing:border-box}@media (prefers-reduced-motion:no-preference){:root{scroll-behavior:smooth}}body{margin:0;font-family:var(--bs-font-sans-serif);font-size:1rem;font-weight:400;line-height:1.5;color:#212529;background-color:#fff;-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent}hr{margin:1rem 0;color:inherit;background-color:currentColor;border:0;opacity:.25}hr:not([size]){height:1px}.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}.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[data-bs-original-title],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:.2em;background-color:#fcf8e3}sub,sup{position:relative;font-size:.75em;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#0d6efd;text-decoration:underline}a:hover{color:#0a58ca}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;direction:ltr;unicode-bidi:bidi-override}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:#d63384;word-wrap:break-word}a>code{color:inherit}kbd{padding:.2rem .4rem;font-size:.875em;color:#fff;background-color:#212529;border-radius:.2rem}kbd kbd{padding:0;font-size:1em;font-weight:700}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:#6c757d;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]::-webkit-calendar-picker-indicator{display:none}[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;font-size:calc(1.275rem + .3vw);line-height:inherit}@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]{outline-offset:-2px;-webkit-appearance:textfield}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-color-swatch-wrapper{padding:0}::file-selector-button{font:inherit}::-webkit-file-upload-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-size:calc(1.625rem + 4.5vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-1{font-size:5rem}}.display-2{font-size:calc(1.575rem + 3.9vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-2{font-size:4.5rem}}.display-3{font-size:calc(1.525rem + 3.3vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-3{font-size:4rem}}.display-4{font-size:calc(1.475rem + 2.7vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-4{font-size:3.5rem}}.display-5{font-size:calc(1.425rem + 2.1vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-5{font-size:3rem}}.display-6{font-size:calc(1.375rem + 1.5vw);font-weight:300;line-height:1.2}@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:#fff;border:1px solid #dee2e6;border-radius:.25rem;max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:.5rem;line-height:1}.figure-caption{font-size:.875em;color:#6c757d}.container,.container-fluid,.container-lg,.container-md,.container-sm,.container-xl,.container-xxl{width:100%;padding-right:var(--bs-gutter-x,.75rem);padding-left:var(--bs-gutter-x,.75rem);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}}.row{--bs-gutter-x:1.5rem;--bs-gutter-y:0;display:flex;flex-wrap:wrap;margin-top:calc(var(--bs-gutter-y) * -1);margin-right:calc(var(--bs-gutter-x) * -.5);margin-left:calc(var(--bs-gutter-x) * -.5)}.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.3333333333%}.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.6666666667%}@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.3333333333%}.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.6666666667%}}@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.3333333333%}.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.6666666667%}}@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.3333333333%}.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.6666666667%}}@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.3333333333%}.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.6666666667%}}@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.3333333333%}.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.6666666667%}}.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-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-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-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-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-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-bg:transparent;--bs-table-accent-bg:transparent;--bs-table-striped-color:#212529;--bs-table-striped-bg:rgba(0, 0, 0, 0.05);--bs-table-active-color:#212529;--bs-table-active-bg:rgba(0, 0, 0, 0.1);--bs-table-hover-color:#212529;--bs-table-hover-bg:rgba(0, 0, 0, 0.075);width:100%;margin-bottom:1rem;color:#212529;vertical-align:top;border-color:#dee2e6}.table>:not(caption)>*>*{padding:.5rem .5rem;background-color:var(--bs-table-bg);border-bottom-width:1px;box-shadow:inset 0 0 0 9999px var(--bs-table-accent-bg)}.table>tbody{vertical-align:inherit}.table>thead{vertical-align:bottom}.table>:not(:last-child)>:last-child>*{border-bottom-color:currentColor}.caption-top{caption-side:top}.table-sm>:not(caption)>*>*{padding:.25rem .25rem}.table-bordered>:not(caption)>*{border-width:1px 0}.table-bordered>:not(caption)>*>*{border-width:0 1px}.table-borderless>:not(caption)>*>*{border-bottom-width:0}.table-striped>tbody>tr:nth-of-type(odd){--bs-table-accent-bg:var(--bs-table-striped-bg);color:var(--bs-table-striped-color)}.table-active{--bs-table-accent-bg:var(--bs-table-active-bg);color:var(--bs-table-active-color)}.table-hover>tbody>tr:hover{--bs-table-accent-bg:var(--bs-table-hover-bg);color:var(--bs-table-hover-color)}.table-primary{--bs-table-bg:#cfe2ff;--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:#000;border-color:#bacbe6}.table-secondary{--bs-table-bg:#e2e3e5;--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:#000;border-color:#cbccce}.table-success{--bs-table-bg:#d1e7dd;--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:#000;border-color:#bcd0c7}.table-info{--bs-table-bg:#cff4fc;--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:#000;border-color:#badce3}.table-warning{--bs-table-bg:#fff3cd;--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:#000;border-color:#e6dbb9}.table-danger{--bs-table-bg:#f8d7da;--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:#000;border-color:#dfc2c4}.table-light{--bs-table-bg:#f8f9fa;--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:#000;border-color:#dfe0e1}.table-dark{--bs-table-bg:#212529;--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:#fff;border-color:#373b3e}.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 + 1px);padding-bottom:calc(.375rem + 1px);margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(.5rem + 1px);padding-bottom:calc(.5rem + 1px);font-size:1.25rem}.col-form-label-sm{padding-top:calc(.25rem + 1px);padding-bottom:calc(.25rem + 1px);font-size:.875rem}.form-text{margin-top:.25rem;font-size:.875em;color:#6c757d}.form-control{display:block;width:100%;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#212529;background-color:#fff;background-clip:padding-box;border:1px solid #ced4da;-webkit-appearance:none;-moz-appearance:none;appearance:none;border-radius:.25rem;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:#212529;background-color:#fff;border-color:#86b7fe;outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.form-control::-webkit-date-and-time-value{height:1.5em}.form-control::-moz-placeholder{color:#6c757d;opacity:1}.form-control::placeholder{color:#6c757d;opacity:1}.form-control:disabled,.form-control[readonly]{background-color:#e9ecef;opacity:1}.form-control::file-selector-button{padding:.375rem .75rem;margin:-.375rem -.75rem;-webkit-margin-end:.75rem;margin-inline-end:.75rem;color:#212529;background-color:#e9ecef;pointer-events:none;border-color:inherit;border-style:solid;border-width:0;border-inline-end-width:1px;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::file-selector-button{transition:none}}.form-control:hover:not(:disabled):not([readonly])::file-selector-button{background-color:#dde0e3}.form-control::-webkit-file-upload-button{padding:.375rem .75rem;margin:-.375rem -.75rem;-webkit-margin-end:.75rem;margin-inline-end:.75rem;color:#212529;background-color:#e9ecef;pointer-events:none;border-color:inherit;border-style:solid;border-width:0;border-inline-end-width:1px;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}@media (prefers-reduced-motion:reduce){.form-control::-webkit-file-upload-button{-webkit-transition:none;transition:none}}.form-control:hover:not(:disabled):not([readonly])::-webkit-file-upload-button{background-color:#dde0e3}.form-control-plaintext{display:block;width:100%;padding:.375rem 0;margin-bottom:0;line-height:1.5;color:#212529;background-color:transparent;border:solid transparent;border-width:1px 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 + 2px));padding:.25rem .5rem;font-size:.875rem;border-radius:.2rem}.form-control-sm::file-selector-button{padding:.25rem .5rem;margin:-.25rem -.5rem;-webkit-margin-end:.5rem;margin-inline-end:.5rem}.form-control-sm::-webkit-file-upload-button{padding:.25rem .5rem;margin:-.25rem -.5rem;-webkit-margin-end:.5rem;margin-inline-end:.5rem}.form-control-lg{min-height:calc(1.5em + (1rem + 2px));padding:.5rem 1rem;font-size:1.25rem;border-radius:.3rem}.form-control-lg::file-selector-button{padding:.5rem 1rem;margin:-.5rem -1rem;-webkit-margin-end:1rem;margin-inline-end:1rem}.form-control-lg::-webkit-file-upload-button{padding:.5rem 1rem;margin:-.5rem -1rem;-webkit-margin-end:1rem;margin-inline-end:1rem}textarea.form-control{min-height:calc(1.5em + (.75rem + 2px))}textarea.form-control-sm{min-height:calc(1.5em + (.5rem + 2px))}textarea.form-control-lg{min-height:calc(1.5em + (1rem + 2px))}.form-control-color{max-width:3rem;height:auto;padding:.375rem}.form-control-color:not(:disabled):not([readonly]){cursor:pointer}.form-control-color::-moz-color-swatch{height:1.5em;border-radius:.25rem}.form-control-color::-webkit-color-swatch{height:1.5em;border-radius:.25rem}.form-select{display:block;width:100%;padding:.375rem 2.25rem .375rem .75rem;-moz-padding-start:calc(0.75rem - 3px);font-size:1rem;font-weight:400;line-height:1.5;color:#212529;background-color:#fff;background-image: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 5l6 6 6-6'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right .75rem center;background-size:16px 12px;border:1px solid #ced4da;border-radius:.25rem;transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out;-webkit-appearance:none;-moz-appearance:none;appearance:none}@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:#e9ecef}.form-select:-moz-focusring{color:transparent;text-shadow:0 0 0 #212529}.form-select-sm{padding-top:.25rem;padding-bottom:.25rem;padding-left:.5rem;font-size:.875rem}.form-select-lg{padding-top:.5rem;padding-bottom:.5rem;padding-left:1rem;font-size:1.25rem}.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-input{width:1em;height:1em;margin-top:.25em;vertical-align:top;background-color:#fff;background-repeat:no-repeat;background-position:center;background-size:contain;border:1px solid rgba(0,0,0,.25);-webkit-appearance:none;-moz-appearance:none;appearance:none;-webkit-print-color-adjust:exact;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]{background-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 10l3 3l6-6'/%3e%3c/svg%3e")}.form-check-input:checked[type=radio]{background-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;background-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{opacity:.5}.form-switch{padding-left:2.5em}.form-switch .form-check-input{width:2em;margin-left:-2.5em;background-image: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");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{background-image: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;background-image: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-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}.form-range{width:100%;height:1.5rem;padding:0;background-color:transparent;-webkit-appearance:none;-moz-appearance:none;appearance:none}.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;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;-webkit-appearance:none;appearance:none}@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:#dee2e6;border-color:transparent;border-radius:1rem}.form-range::-moz-range-thumb{width:1rem;height:1rem;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;-moz-appearance:none;appearance:none}@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:#dee2e6;border-color:transparent;border-radius:1rem}.form-range:disabled{pointer-events:none}.form-range:disabled::-webkit-slider-thumb{background-color:#adb5bd}.form-range:disabled::-moz-range-thumb{background-color:#adb5bd}.form-floating{position:relative}.form-floating>.form-control,.form-floating>.form-select{height:calc(3.5rem + 2px);line-height:1.25}.form-floating>label{position:absolute;top:0;left:0;height:100%;padding:1rem .75rem;pointer-events:none;border:1px 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{padding:1rem .75rem}.form-floating>.form-control::-moz-placeholder{color:transparent}.form-floating>.form-control::placeholder{color:transparent}.form-floating>.form-control:not(:-moz-placeholder-shown){padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control:focus,.form-floating>.form-control:not(:placeholder-shown){padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control:-webkit-autofill{padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-select{padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control:not(:-moz-placeholder-shown)~label{opacity:.65;transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>.form-control:focus~label,.form-floating>.form-control:not(:placeholder-shown)~label,.form-floating>.form-select~label{opacity:.65;transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>.form-control:-webkit-autofill~label{opacity:.65;transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.input-group{position:relative;display:flex;flex-wrap:wrap;align-items:stretch;width:100%}.input-group>.form-control,.input-group>.form-select{position:relative;flex:1 1 auto;width:1%;min-width:0}.input-group>.form-control:focus,.input-group>.form-select:focus{z-index:3}.input-group .btn{position:relative;z-index:2}.input-group .btn:focus{z-index:3}.input-group-text{display:flex;align-items:center;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#212529;text-align:center;white-space:nowrap;background-color:#e9ecef;border:1px solid #ced4da;border-radius:.25rem}.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:.3rem}.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:.2rem}.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)>:not(:last-child):not(.dropdown-toggle):not(.dropdown-menu){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>:nth-last-child(n+3):not(.dropdown-toggle):not(.dropdown-menu){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:-1px;border-top-left-radius:0;border-bottom-left-radius:0}.valid-feedback{display:none;width:100%;margin-top:.25rem;font-size:.875em;color:#198754}.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:rgba(25,135,84,.9);border-radius:.25rem}.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:#198754;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.73L.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.1z'/%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:#198754;box-shadow:0 0 0 .25rem rgba(25,135,84,.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:#198754}.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"]{padding-right:4.125rem;background-image: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 5l6 6 6-6'/%3e%3c/svg%3e"),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.73L.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.1z'/%3e%3c/svg%3e");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:#198754;box-shadow:0 0 0 .25rem rgba(25,135,84,.25)}.form-check-input.is-valid,.was-validated .form-check-input:valid{border-color:#198754}.form-check-input.is-valid:checked,.was-validated .form-check-input:valid:checked{background-color:#198754}.form-check-input.is-valid:focus,.was-validated .form-check-input:valid:focus{box-shadow:0 0 0 .25rem rgba(25,135,84,.25)}.form-check-input.is-valid~.form-check-label,.was-validated .form-check-input:valid~.form-check-label{color:#198754}.form-check-inline .form-check-input~.valid-feedback{margin-left:.5em}.input-group .form-control.is-valid,.input-group .form-select.is-valid,.was-validated .input-group .form-control:valid,.was-validated .input-group .form-select:valid{z-index:1}.input-group .form-control.is-valid:focus,.input-group .form-select.is-valid:focus,.was-validated .input-group .form-control:valid:focus,.was-validated .input-group .form-select:valid:focus{z-index:3}.invalid-feedback{display:none;width:100%;margin-top:.25rem;font-size:.875em;color:#dc3545}.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:rgba(220,53,69,.9);border-radius:.25rem}.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:#dc3545;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:#dc3545;box-shadow:0 0 0 .25rem rgba(220,53,69,.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:#dc3545}.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"]{padding-right:4.125rem;background-image: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 5l6 6 6-6'/%3e%3c/svg%3e"),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-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:#dc3545;box-shadow:0 0 0 .25rem rgba(220,53,69,.25)}.form-check-input.is-invalid,.was-validated .form-check-input:invalid{border-color:#dc3545}.form-check-input.is-invalid:checked,.was-validated .form-check-input:invalid:checked{background-color:#dc3545}.form-check-input.is-invalid:focus,.was-validated .form-check-input:invalid:focus{box-shadow:0 0 0 .25rem rgba(220,53,69,.25)}.form-check-input.is-invalid~.form-check-label,.was-validated .form-check-input:invalid~.form-check-label{color:#dc3545}.form-check-inline .form-check-input~.invalid-feedback{margin-left:.5em}.input-group .form-control.is-invalid,.input-group .form-select.is-invalid,.was-validated .input-group .form-control:invalid,.was-validated .input-group .form-select:invalid{z-index:2}.input-group .form-control.is-invalid:focus,.input-group .form-select.is-invalid:focus,.was-validated .input-group .form-control:invalid:focus,.was-validated .input-group .form-select:invalid:focus{z-index:3}.btn{display:inline-block;font-weight:400;line-height:1.5;color:#212529;text-align:center;text-decoration:none;vertical-align:middle;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;user-select:none;background-color:transparent;border:1px solid transparent;padding:.375rem .75rem;font-size:1rem;border-radius:.25rem;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:#212529}.btn-check:focus+.btn,.btn:focus{outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.btn.disabled,.btn:disabled,fieldset:disabled .btn{pointer-events:none;opacity:.65}.btn-primary{color:#fff;background-color:#0d6efd;border-color:#0d6efd}.btn-primary:hover{color:#fff;background-color:#0b5ed7;border-color:#0a58ca}.btn-check:focus+.btn-primary,.btn-primary:focus{color:#fff;background-color:#0b5ed7;border-color:#0a58ca;box-shadow:0 0 0 .25rem rgba(49,132,253,.5)}.btn-check:active+.btn-primary,.btn-check:checked+.btn-primary,.btn-primary.active,.btn-primary:active,.show>.btn-primary.dropdown-toggle{color:#fff;background-color:#0a58ca;border-color:#0a53be}.btn-check:active+.btn-primary:focus,.btn-check:checked+.btn-primary:focus,.btn-primary.active:focus,.btn-primary:active:focus,.show>.btn-primary.dropdown-toggle:focus{box-shadow:0 0 0 .25rem rgba(49,132,253,.5)}.btn-primary.disabled,.btn-primary:disabled{color:#fff;background-color:#0d6efd;border-color:#0d6efd}.btn-secondary{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-secondary:hover{color:#fff;background-color:#5c636a;border-color:#565e64}.btn-check:focus+.btn-secondary,.btn-secondary:focus{color:#fff;background-color:#5c636a;border-color:#565e64;box-shadow:0 0 0 .25rem rgba(130,138,145,.5)}.btn-check:active+.btn-secondary,.btn-check:checked+.btn-secondary,.btn-secondary.active,.btn-secondary:active,.show>.btn-secondary.dropdown-toggle{color:#fff;background-color:#565e64;border-color:#51585e}.btn-check:active+.btn-secondary:focus,.btn-check:checked+.btn-secondary:focus,.btn-secondary.active:focus,.btn-secondary:active:focus,.show>.btn-secondary.dropdown-toggle:focus{box-shadow:0 0 0 .25rem rgba(130,138,145,.5)}.btn-secondary.disabled,.btn-secondary:disabled{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-success{color:#fff;background-color:#198754;border-color:#198754}.btn-success:hover{color:#fff;background-color:#157347;border-color:#146c43}.btn-check:focus+.btn-success,.btn-success:focus{color:#fff;background-color:#157347;border-color:#146c43;box-shadow:0 0 0 .25rem rgba(60,153,110,.5)}.btn-check:active+.btn-success,.btn-check:checked+.btn-success,.btn-success.active,.btn-success:active,.show>.btn-success.dropdown-toggle{color:#fff;background-color:#146c43;border-color:#13653f}.btn-check:active+.btn-success:focus,.btn-check:checked+.btn-success:focus,.btn-success.active:focus,.btn-success:active:focus,.show>.btn-success.dropdown-toggle:focus{box-shadow:0 0 0 .25rem rgba(60,153,110,.5)}.btn-success.disabled,.btn-success:disabled{color:#fff;background-color:#198754;border-color:#198754}.btn-info{color:#000;background-color:#0dcaf0;border-color:#0dcaf0}.btn-info:hover{color:#000;background-color:#31d2f2;border-color:#25cff2}.btn-check:focus+.btn-info,.btn-info:focus{color:#000;background-color:#31d2f2;border-color:#25cff2;box-shadow:0 0 0 .25rem rgba(11,172,204,.5)}.btn-check:active+.btn-info,.btn-check:checked+.btn-info,.btn-info.active,.btn-info:active,.show>.btn-info.dropdown-toggle{color:#000;background-color:#3dd5f3;border-color:#25cff2}.btn-check:active+.btn-info:focus,.btn-check:checked+.btn-info:focus,.btn-info.active:focus,.btn-info:active:focus,.show>.btn-info.dropdown-toggle:focus{box-shadow:0 0 0 .25rem rgba(11,172,204,.5)}.btn-info.disabled,.btn-info:disabled{color:#000;background-color:#0dcaf0;border-color:#0dcaf0}.btn-warning{color:#000;background-color:#ffc107;border-color:#ffc107}.btn-warning:hover{color:#000;background-color:#ffca2c;border-color:#ffc720}.btn-check:focus+.btn-warning,.btn-warning:focus{color:#000;background-color:#ffca2c;border-color:#ffc720;box-shadow:0 0 0 .25rem rgba(217,164,6,.5)}.btn-check:active+.btn-warning,.btn-check:checked+.btn-warning,.btn-warning.active,.btn-warning:active,.show>.btn-warning.dropdown-toggle{color:#000;background-color:#ffcd39;border-color:#ffc720}.btn-check:active+.btn-warning:focus,.btn-check:checked+.btn-warning:focus,.btn-warning.active:focus,.btn-warning:active:focus,.show>.btn-warning.dropdown-toggle:focus{box-shadow:0 0 0 .25rem rgba(217,164,6,.5)}.btn-warning.disabled,.btn-warning:disabled{color:#000;background-color:#ffc107;border-color:#ffc107}.btn-danger{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-danger:hover{color:#fff;background-color:#bb2d3b;border-color:#b02a37}.btn-check:focus+.btn-danger,.btn-danger:focus{color:#fff;background-color:#bb2d3b;border-color:#b02a37;box-shadow:0 0 0 .25rem rgba(225,83,97,.5)}.btn-check:active+.btn-danger,.btn-check:checked+.btn-danger,.btn-danger.active,.btn-danger:active,.show>.btn-danger.dropdown-toggle{color:#fff;background-color:#b02a37;border-color:#a52834}.btn-check:active+.btn-danger:focus,.btn-check:checked+.btn-danger:focus,.btn-danger.active:focus,.btn-danger:active:focus,.show>.btn-danger.dropdown-toggle:focus{box-shadow:0 0 0 .25rem rgba(225,83,97,.5)}.btn-danger.disabled,.btn-danger:disabled{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-light{color:#000;background-color:#f8f9fa;border-color:#f8f9fa}.btn-light:hover{color:#000;background-color:#f9fafb;border-color:#f9fafb}.btn-check:focus+.btn-light,.btn-light:focus{color:#000;background-color:#f9fafb;border-color:#f9fafb;box-shadow:0 0 0 .25rem rgba(211,212,213,.5)}.btn-check:active+.btn-light,.btn-check:checked+.btn-light,.btn-light.active,.btn-light:active,.show>.btn-light.dropdown-toggle{color:#000;background-color:#f9fafb;border-color:#f9fafb}.btn-check:active+.btn-light:focus,.btn-check:checked+.btn-light:focus,.btn-light.active:focus,.btn-light:active:focus,.show>.btn-light.dropdown-toggle:focus{box-shadow:0 0 0 .25rem rgba(211,212,213,.5)}.btn-light.disabled,.btn-light:disabled{color:#000;background-color:#f8f9fa;border-color:#f8f9fa}.btn-dark{color:#fff;background-color:#212529;border-color:#212529}.btn-dark:hover{color:#fff;background-color:#1c1f23;border-color:#1a1e21}.btn-check:focus+.btn-dark,.btn-dark:focus{color:#fff;background-color:#1c1f23;border-color:#1a1e21;box-shadow:0 0 0 .25rem rgba(66,70,73,.5)}.btn-check:active+.btn-dark,.btn-check:checked+.btn-dark,.btn-dark.active,.btn-dark:active,.show>.btn-dark.dropdown-toggle{color:#fff;background-color:#1a1e21;border-color:#191c1f}.btn-check:active+.btn-dark:focus,.btn-check:checked+.btn-dark:focus,.btn-dark.active:focus,.btn-dark:active:focus,.show>.btn-dark.dropdown-toggle:focus{box-shadow:0 0 0 .25rem rgba(66,70,73,.5)}.btn-dark.disabled,.btn-dark:disabled{color:#fff;background-color:#212529;border-color:#212529}.btn-outline-primary{color:#0d6efd;border-color:#0d6efd}.btn-outline-primary:hover{color:#fff;background-color:#0d6efd;border-color:#0d6efd}.btn-check:focus+.btn-outline-primary,.btn-outline-primary:focus{box-shadow:0 0 0 .25rem rgba(13,110,253,.5)}.btn-check:active+.btn-outline-primary,.btn-check:checked+.btn-outline-primary,.btn-outline-primary.active,.btn-outline-primary.dropdown-toggle.show,.btn-outline-primary:active{color:#fff;background-color:#0d6efd;border-color:#0d6efd}.btn-check:active+.btn-outline-primary:focus,.btn-check:checked+.btn-outline-primary:focus,.btn-outline-primary.active:focus,.btn-outline-primary.dropdown-toggle.show:focus,.btn-outline-primary:active:focus{box-shadow:0 0 0 .25rem rgba(13,110,253,.5)}.btn-outline-primary.disabled,.btn-outline-primary:disabled{color:#0d6efd;background-color:transparent}.btn-outline-secondary{color:#6c757d;border-color:#6c757d}.btn-outline-secondary:hover{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-check:focus+.btn-outline-secondary,.btn-outline-secondary:focus{box-shadow:0 0 0 .25rem rgba(108,117,125,.5)}.btn-check:active+.btn-outline-secondary,.btn-check:checked+.btn-outline-secondary,.btn-outline-secondary.active,.btn-outline-secondary.dropdown-toggle.show,.btn-outline-secondary:active{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-check:active+.btn-outline-secondary:focus,.btn-check:checked+.btn-outline-secondary:focus,.btn-outline-secondary.active:focus,.btn-outline-secondary.dropdown-toggle.show:focus,.btn-outline-secondary:active:focus{box-shadow:0 0 0 .25rem rgba(108,117,125,.5)}.btn-outline-secondary.disabled,.btn-outline-secondary:disabled{color:#6c757d;background-color:transparent}.btn-outline-success{color:#198754;border-color:#198754}.btn-outline-success:hover{color:#fff;background-color:#198754;border-color:#198754}.btn-check:focus+.btn-outline-success,.btn-outline-success:focus{box-shadow:0 0 0 .25rem rgba(25,135,84,.5)}.btn-check:active+.btn-outline-success,.btn-check:checked+.btn-outline-success,.btn-outline-success.active,.btn-outline-success.dropdown-toggle.show,.btn-outline-success:active{color:#fff;background-color:#198754;border-color:#198754}.btn-check:active+.btn-outline-success:focus,.btn-check:checked+.btn-outline-success:focus,.btn-outline-success.active:focus,.btn-outline-success.dropdown-toggle.show:focus,.btn-outline-success:active:focus{box-shadow:0 0 0 .25rem rgba(25,135,84,.5)}.btn-outline-success.disabled,.btn-outline-success:disabled{color:#198754;background-color:transparent}.btn-outline-info{color:#0dcaf0;border-color:#0dcaf0}.btn-outline-info:hover{color:#000;background-color:#0dcaf0;border-color:#0dcaf0}.btn-check:focus+.btn-outline-info,.btn-outline-info:focus{box-shadow:0 0 0 .25rem rgba(13,202,240,.5)}.btn-check:active+.btn-outline-info,.btn-check:checked+.btn-outline-info,.btn-outline-info.active,.btn-outline-info.dropdown-toggle.show,.btn-outline-info:active{color:#000;background-color:#0dcaf0;border-color:#0dcaf0}.btn-check:active+.btn-outline-info:focus,.btn-check:checked+.btn-outline-info:focus,.btn-outline-info.active:focus,.btn-outline-info.dropdown-toggle.show:focus,.btn-outline-info:active:focus{box-shadow:0 0 0 .25rem rgba(13,202,240,.5)}.btn-outline-info.disabled,.btn-outline-info:disabled{color:#0dcaf0;background-color:transparent}.btn-outline-warning{color:#ffc107;border-color:#ffc107}.btn-outline-warning:hover{color:#000;background-color:#ffc107;border-color:#ffc107}.btn-check:focus+.btn-outline-warning,.btn-outline-warning:focus{box-shadow:0 0 0 .25rem rgba(255,193,7,.5)}.btn-check:active+.btn-outline-warning,.btn-check:checked+.btn-outline-warning,.btn-outline-warning.active,.btn-outline-warning.dropdown-toggle.show,.btn-outline-warning:active{color:#000;background-color:#ffc107;border-color:#ffc107}.btn-check:active+.btn-outline-warning:focus,.btn-check:checked+.btn-outline-warning:focus,.btn-outline-warning.active:focus,.btn-outline-warning.dropdown-toggle.show:focus,.btn-outline-warning:active:focus{box-shadow:0 0 0 .25rem rgba(255,193,7,.5)}.btn-outline-warning.disabled,.btn-outline-warning:disabled{color:#ffc107;background-color:transparent}.btn-outline-danger{color:#dc3545;border-color:#dc3545}.btn-outline-danger:hover{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-check:focus+.btn-outline-danger,.btn-outline-danger:focus{box-shadow:0 0 0 .25rem rgba(220,53,69,.5)}.btn-check:active+.btn-outline-danger,.btn-check:checked+.btn-outline-danger,.btn-outline-danger.active,.btn-outline-danger.dropdown-toggle.show,.btn-outline-danger:active{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-check:active+.btn-outline-danger:focus,.btn-check:checked+.btn-outline-danger:focus,.btn-outline-danger.active:focus,.btn-outline-danger.dropdown-toggle.show:focus,.btn-outline-danger:active:focus{box-shadow:0 0 0 .25rem rgba(220,53,69,.5)}.btn-outline-danger.disabled,.btn-outline-danger:disabled{color:#dc3545;background-color:transparent}.btn-outline-light{color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light:hover{color:#000;background-color:#f8f9fa;border-color:#f8f9fa}.btn-check:focus+.btn-outline-light,.btn-outline-light:focus{box-shadow:0 0 0 .25rem rgba(248,249,250,.5)}.btn-check:active+.btn-outline-light,.btn-check:checked+.btn-outline-light,.btn-outline-light.active,.btn-outline-light.dropdown-toggle.show,.btn-outline-light:active{color:#000;background-color:#f8f9fa;border-color:#f8f9fa}.btn-check:active+.btn-outline-light:focus,.btn-check:checked+.btn-outline-light:focus,.btn-outline-light.active:focus,.btn-outline-light.dropdown-toggle.show:focus,.btn-outline-light:active:focus{box-shadow:0 0 0 .25rem rgba(248,249,250,.5)}.btn-outline-light.disabled,.btn-outline-light:disabled{color:#f8f9fa;background-color:transparent}.btn-outline-dark{color:#212529;border-color:#212529}.btn-outline-dark:hover{color:#fff;background-color:#212529;border-color:#212529}.btn-check:focus+.btn-outline-dark,.btn-outline-dark:focus{box-shadow:0 0 0 .25rem rgba(33,37,41,.5)}.btn-check:active+.btn-outline-dark,.btn-check:checked+.btn-outline-dark,.btn-outline-dark.active,.btn-outline-dark.dropdown-toggle.show,.btn-outline-dark:active{color:#fff;background-color:#212529;border-color:#212529}.btn-check:active+.btn-outline-dark:focus,.btn-check:checked+.btn-outline-dark:focus,.btn-outline-dark.active:focus,.btn-outline-dark.dropdown-toggle.show:focus,.btn-outline-dark:active:focus{box-shadow:0 0 0 .25rem rgba(33,37,41,.5)}.btn-outline-dark.disabled,.btn-outline-dark:disabled{color:#212529;background-color:transparent}.btn-link{font-weight:400;color:#0d6efd;text-decoration:underline}.btn-link:hover{color:#0a58ca}.btn-link.disabled,.btn-link:disabled{color:#6c757d}.btn-group-lg>.btn,.btn-lg{padding:.5rem 1rem;font-size:1.25rem;border-radius:.3rem}.btn-group-sm>.btn,.btn-sm{padding:.25rem .5rem;font-size:.875rem;border-radius:.2rem}.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}}.dropdown,.dropend,.dropstart,.dropup{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{position:absolute;z-index:1000;display:none;min-width:10rem;padding:.5rem 0;margin:0;font-size:1rem;color:#212529;text-align:left;list-style:none;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.15);border-radius:.25rem}.dropdown-menu[data-bs-popper]{top:100%;left:0;margin-top:.125rem}.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:.125rem}.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:.125rem}.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:.125rem}.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:.5rem 0;overflow:hidden;border-top:1px solid rgba(0,0,0,.15)}.dropdown-item{display:block;width:100%;padding:.25rem 1rem;clear:both;font-weight:400;color:#212529;text-align:inherit;text-decoration:none;white-space:nowrap;background-color:transparent;border:0}.dropdown-item:focus,.dropdown-item:hover{color:#1e2125;background-color:#e9ecef}.dropdown-item.active,.dropdown-item:active{color:#fff;text-decoration:none;background-color:#0d6efd}.dropdown-item.disabled,.dropdown-item:disabled{color:#adb5bd;pointer-events:none;background-color:transparent}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:.5rem 1rem;margin-bottom:0;font-size:.875rem;color:#6c757d;white-space:nowrap}.dropdown-item-text{display:block;padding:.25rem 1rem;color:#212529}.dropdown-menu-dark{color:#dee2e6;background-color:#343a40;border-color:rgba(0,0,0,.15)}.dropdown-menu-dark .dropdown-item{color:#dee2e6}.dropdown-menu-dark .dropdown-item:focus,.dropdown-menu-dark .dropdown-item:hover{color:#fff;background-color:rgba(255,255,255,.15)}.dropdown-menu-dark .dropdown-item.active,.dropdown-menu-dark .dropdown-item:active{color:#fff;background-color:#0d6efd}.dropdown-menu-dark .dropdown-item.disabled,.dropdown-menu-dark .dropdown-item:disabled{color:#adb5bd}.dropdown-menu-dark .dropdown-divider{border-color:rgba(0,0,0,.15)}.dropdown-menu-dark .dropdown-item-text{color:#dee2e6}.dropdown-menu-dark .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>.btn-group:not(:first-child),.btn-group>.btn:not(:first-child){margin-left:-1px}.btn-group>.btn-group:not(:last-child)>.btn,.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:-1px}.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~.btn{border-top-left-radius:0;border-top-right-radius:0}.nav{display:flex;flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:.5rem 1rem;color:#0d6efd;text-decoration:none;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:#0a58ca}.nav-link.disabled{color:#6c757d;pointer-events:none;cursor:default}.nav-tabs{border-bottom:1px solid #dee2e6}.nav-tabs .nav-link{margin-bottom:-1px;background:0 0;border:1px solid transparent;border-top-left-radius:.25rem;border-top-right-radius:.25rem}.nav-tabs .nav-link:focus,.nav-tabs .nav-link:hover{border-color:#e9ecef #e9ecef #dee2e6;isolation:isolate}.nav-tabs .nav-link.disabled{color:#6c757d;background-color:transparent;border-color:transparent}.nav-tabs .nav-item.show .nav-link,.nav-tabs .nav-link.active{color:#495057;background-color:#fff;border-color:#dee2e6 #dee2e6 #fff}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.nav-pills .nav-link{background:0 0;border:0;border-radius:.25rem}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:#fff;background-color:#0d6efd}.nav-fill .nav-item,.nav-fill>.nav-link{flex:1 1 auto;text-align:center}.nav-justified .nav-item,.nav-justified>.nav-link{flex-basis:0;flex-grow:1;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{position:relative;display:flex;flex-wrap:wrap;align-items:center;justify-content:space-between;padding-top:.5rem;padding-bottom:.5rem}.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:.3125rem;padding-bottom:.3125rem;margin-right:1rem;font-size:1.25rem;text-decoration:none;white-space:nowrap}.navbar-nav{display:flex;flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link{padding-right:0;padding-left:0}.navbar-nav .dropdown-menu{position:static}.navbar-text{padding-top:.5rem;padding-bottom:.5rem}.navbar-collapse{flex-basis:100%;flex-grow:1;align-items:center}.navbar-toggler{padding:.25rem .75rem;font-size:1.25rem;line-height:1;background-color:transparent;border:1px solid transparent;border-radius:.25rem;transition:box-shadow .15s ease-in-out}@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 .25rem}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;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:.5rem;padding-left:.5rem}.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}}@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:.5rem;padding-left:.5rem}.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}}@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:.5rem;padding-left:.5rem}.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}}@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:.5rem;padding-left:.5rem}.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}}@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:.5rem;padding-left:.5rem}.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{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:.5rem;padding-left:.5rem}.navbar-expand .navbar-nav-scroll{overflow:visible}.navbar-expand .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-light .navbar-brand{color:rgba(0,0,0,.9)}.navbar-light .navbar-brand:focus,.navbar-light .navbar-brand:hover{color:rgba(0,0,0,.9)}.navbar-light .navbar-nav .nav-link{color:rgba(0,0,0,.55)}.navbar-light .navbar-nav .nav-link:focus,.navbar-light .navbar-nav .nav-link:hover{color:rgba(0,0,0,.7)}.navbar-light .navbar-nav .nav-link.disabled{color:rgba(0,0,0,.3)}.navbar-light .navbar-nav .nav-link.active,.navbar-light .navbar-nav .show>.nav-link{color:rgba(0,0,0,.9)}.navbar-light .navbar-toggler{color:rgba(0,0,0,.55);border-color:rgba(0,0,0,.1)}.navbar-light .navbar-toggler-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%280, 0, 0, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.navbar-light .navbar-text{color:rgba(0,0,0,.55)}.navbar-light .navbar-text a,.navbar-light .navbar-text a:focus,.navbar-light .navbar-text a:hover{color:rgba(0,0,0,.9)}.navbar-dark .navbar-brand{color:#fff}.navbar-dark .navbar-brand:focus,.navbar-dark .navbar-brand:hover{color:#fff}.navbar-dark .navbar-nav .nav-link{color:rgba(255,255,255,.55)}.navbar-dark .navbar-nav .nav-link:focus,.navbar-dark .navbar-nav .nav-link:hover{color:rgba(255,255,255,.75)}.navbar-dark .navbar-nav .nav-link.disabled{color:rgba(255,255,255,.25)}.navbar-dark .navbar-nav .nav-link.active,.navbar-dark .navbar-nav .show>.nav-link{color:#fff}.navbar-dark .navbar-toggler{color:rgba(255,255,255,.55);border-color:rgba(255,255,255,.1)}.navbar-dark .navbar-toggler-icon{background-image: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")}.navbar-dark .navbar-text{color:rgba(255,255,255,.55)}.navbar-dark .navbar-text a,.navbar-dark .navbar-text a:focus,.navbar-dark .navbar-text a:hover{color:#fff}.card{position:relative;display:flex;flex-direction:column;min-width:0;word-wrap:break-word;background-color:#fff;background-clip:border-box;border:1px solid rgba(0,0,0,.125);border-radius:.25rem}.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:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.card>.list-group:last-child{border-bottom-width:0;border-bottom-right-radius:calc(.25rem - 1px);border-bottom-left-radius:calc(.25rem - 1px)}.card>.card-header+.list-group,.card>.list-group+.card-footer{border-top:0}.card-body{flex:1 1 auto;padding:1rem 1rem}.card-title{margin-bottom:.5rem}.card-subtitle{margin-top:-.25rem;margin-bottom:0}.card-text:last-child{margin-bottom:0}.card-link:hover{text-decoration:none}.card-link+.card-link{margin-left:1rem}.card-header{padding:.5rem 1rem;margin-bottom:0;background-color:rgba(0,0,0,.03);border-bottom:1px solid rgba(0,0,0,.125)}.card-header:first-child{border-radius:calc(.25rem - 1px) calc(.25rem - 1px) 0 0}.card-footer{padding:.5rem 1rem;background-color:rgba(0,0,0,.03);border-top:1px solid rgba(0,0,0,.125)}.card-footer:last-child{border-radius:0 0 calc(.25rem - 1px) calc(.25rem - 1px)}.card-header-tabs{margin-right:-.5rem;margin-bottom:-.5rem;margin-left:-.5rem;border-bottom:0}.card-header-pills{margin-right:-.5rem;margin-left:-.5rem}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:1rem;border-radius:calc(.25rem - 1px)}.card-img,.card-img-bottom,.card-img-top{width:100%}.card-img,.card-img-top{border-top-left-radius:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.card-img,.card-img-bottom{border-bottom-right-radius:calc(.25rem - 1px);border-bottom-left-radius:calc(.25rem - 1px)}.card-group>.card{margin-bottom:.75rem}@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-button{position:relative;display:flex;align-items:center;width:100%;padding:1rem 1.25rem;font-size:1rem;color:#212529;text-align:left;background-color:#fff;border:0;border-radius:0;overflow-anchor:none;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out,border-radius .15s ease}@media (prefers-reduced-motion:reduce){.accordion-button{transition:none}}.accordion-button:not(.collapsed){color:#0c63e4;background-color:#e7f1ff;box-shadow:inset 0 -1px 0 rgba(0,0,0,.125)}.accordion-button:not(.collapsed)::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%230c63e4'%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-.708z'/%3e%3c/svg%3e");transform:rotate(-180deg)}.accordion-button::after{flex-shrink:0;width:1.25rem;height:1.25rem;margin-left:auto;content:"";background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23212529'%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-.708z'/%3e%3c/svg%3e");background-repeat:no-repeat;background-size:1.25rem;transition:transform .2s ease-in-out}@media (prefers-reduced-motion:reduce){.accordion-button::after{transition:none}}.accordion-button:hover{z-index:2}.accordion-button:focus{z-index:3;border-color:#86b7fe;outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.accordion-header{margin-bottom:0}.accordion-item{background-color:#fff;border:1px solid rgba(0,0,0,.125)}.accordion-item:first-of-type{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.accordion-item:first-of-type .accordion-button{border-top-left-radius:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.accordion-item:not(:first-of-type){border-top:0}.accordion-item:last-of-type{border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.accordion-item:last-of-type .accordion-button.collapsed{border-bottom-right-radius:calc(.25rem - 1px);border-bottom-left-radius:calc(.25rem - 1px)}.accordion-item:last-of-type .accordion-collapse{border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.accordion-body{padding:1rem 1.25rem}.accordion-flush .accordion-collapse{border-width:0}.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-button{border-radius:0}.breadcrumb{display:flex;flex-wrap:wrap;padding:0 0;margin-bottom:1rem;list-style:none}.breadcrumb-item+.breadcrumb-item{padding-left:.5rem}.breadcrumb-item+.breadcrumb-item::before{float:left;padding-right:.5rem;color:#6c757d;content:var(--bs-breadcrumb-divider, "/")}.breadcrumb-item.active{color:#6c757d}.pagination{display:flex;padding-left:0;list-style:none}.page-link{position:relative;display:block;color:#0d6efd;text-decoration:none;background-color:#fff;border:1px solid #dee2e6;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:#0a58ca;background-color:#e9ecef;border-color:#dee2e6}.page-link:focus{z-index:3;color:#0a58ca;background-color:#e9ecef;outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.page-item:not(:first-child) .page-link{margin-left:-1px}.page-item.active .page-link{z-index:3;color:#fff;background-color:#0d6efd;border-color:#0d6efd}.page-item.disabled .page-link{color:#6c757d;pointer-events:none;background-color:#fff;border-color:#dee2e6}.page-link{padding:.375rem .75rem}.page-item:first-child .page-link{border-top-left-radius:.25rem;border-bottom-left-radius:.25rem}.page-item:last-child .page-link{border-top-right-radius:.25rem;border-bottom-right-radius:.25rem}.pagination-lg .page-link{padding:.75rem 1.5rem;font-size:1.25rem}.pagination-lg .page-item:first-child .page-link{border-top-left-radius:.3rem;border-bottom-left-radius:.3rem}.pagination-lg .page-item:last-child .page-link{border-top-right-radius:.3rem;border-bottom-right-radius:.3rem}.pagination-sm .page-link{padding:.25rem .5rem;font-size:.875rem}.pagination-sm .page-item:first-child .page-link{border-top-left-radius:.2rem;border-bottom-left-radius:.2rem}.pagination-sm .page-item:last-child .page-link{border-top-right-radius:.2rem;border-bottom-right-radius:.2rem}.badge{display:inline-block;padding:.35em .65em;font-size:.75em;font-weight:700;line-height:1;color:#fff;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:.25rem}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.alert{position:relative;padding:1rem 1rem;margin-bottom:1rem;border:1px solid transparent;border-radius:.25rem}.alert-heading{color:inherit}.alert-link{font-weight:700}.alert-dismissible{padding-right:3rem}.alert-dismissible .btn-close{position:absolute;top:0;right:0;z-index:2;padding:1.25rem 1rem}.alert-primary{color:#084298;background-color:#cfe2ff;border-color:#b6d4fe}.alert-primary .alert-link{color:#06357a}.alert-secondary{color:#41464b;background-color:#e2e3e5;border-color:#d3d6d8}.alert-secondary .alert-link{color:#34383c}.alert-success{color:#0f5132;background-color:#d1e7dd;border-color:#badbcc}.alert-success .alert-link{color:#0c4128}.alert-info{color:#055160;background-color:#cff4fc;border-color:#b6effb}.alert-info .alert-link{color:#04414d}.alert-warning{color:#664d03;background-color:#fff3cd;border-color:#ffecb5}.alert-warning .alert-link{color:#523e02}.alert-danger{color:#842029;background-color:#f8d7da;border-color:#f5c2c7}.alert-danger .alert-link{color:#6a1a21}.alert-light{color:#636464;background-color:#fefefe;border-color:#fdfdfe}.alert-light .alert-link{color:#4f5050}.alert-dark{color:#141619;background-color:#d3d3d4;border-color:#bcbebf}.alert-dark .alert-link{color:#101214}@-webkit-keyframes progress-bar-stripes{0%{background-position-x:1rem}}@keyframes progress-bar-stripes{0%{background-position-x:1rem}}.progress{display:flex;height:1rem;overflow:hidden;font-size:.75rem;background-color:#e9ecef;border-radius:.25rem}.progress-bar{display:flex;flex-direction:column;justify-content:center;overflow:hidden;color:#fff;text-align:center;white-space:nowrap;background-color:#0d6efd;transition:width .6s ease}@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:1rem 1rem}.progress-bar-animated{-webkit-animation:1s linear infinite progress-bar-stripes;animation:1s linear infinite progress-bar-stripes}@media (prefers-reduced-motion:reduce){.progress-bar-animated{-webkit-animation:none;animation:none}}.list-group{display:flex;flex-direction:column;padding-left:0;margin-bottom:0;border-radius:.25rem}.list-group-numbered{list-style-type:none;counter-reset:section}.list-group-numbered>li::before{content:counters(section, ".") ". ";counter-increment:section}.list-group-item-action{width:100%;color:#495057;text-align:inherit}.list-group-item-action:focus,.list-group-item-action:hover{z-index:1;color:#495057;text-decoration:none;background-color:#f8f9fa}.list-group-item-action:active{color:#212529;background-color:#e9ecef}.list-group-item{position:relative;display:block;padding:.5rem 1rem;color:#212529;text-decoration:none;background-color:#fff;border:1px solid rgba(0,0,0,.125)}.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:#6c757d;pointer-events:none;background-color:#fff}.list-group-item.active{z-index:2;color:#fff;background-color:#0d6efd;border-color:#0d6efd}.list-group-item+.list-group-item{border-top-width:0}.list-group-item+.list-group-item.active{margin-top:-1px;border-top-width:1px}.list-group-horizontal{flex-direction:row}.list-group-horizontal>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal>.list-group-item:last-child{border-top-right-radius:.25rem;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:1px;border-left-width:0}.list-group-horizontal>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}@media (min-width:576px){.list-group-horizontal-sm{flex-direction:row}.list-group-horizontal-sm>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-sm>.list-group-item:last-child{border-top-right-radius:.25rem;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:1px;border-left-width:0}.list-group-horizontal-sm>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:768px){.list-group-horizontal-md{flex-direction:row}.list-group-horizontal-md>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-md>.list-group-item:last-child{border-top-right-radius:.25rem;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:1px;border-left-width:0}.list-group-horizontal-md>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:992px){.list-group-horizontal-lg{flex-direction:row}.list-group-horizontal-lg>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-lg>.list-group-item:last-child{border-top-right-radius:.25rem;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:1px;border-left-width:0}.list-group-horizontal-lg>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:1200px){.list-group-horizontal-xl{flex-direction:row}.list-group-horizontal-xl>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-xl>.list-group-item:last-child{border-top-right-radius:.25rem;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:1px;border-left-width:0}.list-group-horizontal-xl>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:1400px){.list-group-horizontal-xxl{flex-direction:row}.list-group-horizontal-xxl>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-xxl>.list-group-item:last-child{border-top-right-radius:.25rem;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:1px;border-left-width:0}.list-group-horizontal-xxl>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}.list-group-flush{border-radius:0}.list-group-flush>.list-group-item{border-width:0 0 1px}.list-group-flush>.list-group-item:last-child{border-bottom-width:0}.list-group-item-primary{color:#084298;background-color:#cfe2ff}.list-group-item-primary.list-group-item-action:focus,.list-group-item-primary.list-group-item-action:hover{color:#084298;background-color:#bacbe6}.list-group-item-primary.list-group-item-action.active{color:#fff;background-color:#084298;border-color:#084298}.list-group-item-secondary{color:#41464b;background-color:#e2e3e5}.list-group-item-secondary.list-group-item-action:focus,.list-group-item-secondary.list-group-item-action:hover{color:#41464b;background-color:#cbccce}.list-group-item-secondary.list-group-item-action.active{color:#fff;background-color:#41464b;border-color:#41464b}.list-group-item-success{color:#0f5132;background-color:#d1e7dd}.list-group-item-success.list-group-item-action:focus,.list-group-item-success.list-group-item-action:hover{color:#0f5132;background-color:#bcd0c7}.list-group-item-success.list-group-item-action.active{color:#fff;background-color:#0f5132;border-color:#0f5132}.list-group-item-info{color:#055160;background-color:#cff4fc}.list-group-item-info.list-group-item-action:focus,.list-group-item-info.list-group-item-action:hover{color:#055160;background-color:#badce3}.list-group-item-info.list-group-item-action.active{color:#fff;background-color:#055160;border-color:#055160}.list-group-item-warning{color:#664d03;background-color:#fff3cd}.list-group-item-warning.list-group-item-action:focus,.list-group-item-warning.list-group-item-action:hover{color:#664d03;background-color:#e6dbb9}.list-group-item-warning.list-group-item-action.active{color:#fff;background-color:#664d03;border-color:#664d03}.list-group-item-danger{color:#842029;background-color:#f8d7da}.list-group-item-danger.list-group-item-action:focus,.list-group-item-danger.list-group-item-action:hover{color:#842029;background-color:#dfc2c4}.list-group-item-danger.list-group-item-action.active{color:#fff;background-color:#842029;border-color:#842029}.list-group-item-light{color:#636464;background-color:#fefefe}.list-group-item-light.list-group-item-action:focus,.list-group-item-light.list-group-item-action:hover{color:#636464;background-color:#e5e5e5}.list-group-item-light.list-group-item-action.active{color:#fff;background-color:#636464;border-color:#636464}.list-group-item-dark{color:#141619;background-color:#d3d3d4}.list-group-item-dark.list-group-item-action:focus,.list-group-item-dark.list-group-item-action:hover{color:#141619;background-color:#bebebf}.list-group-item-dark.list-group-item-action.active{color:#fff;background-color:#141619;border-color:#141619}.btn-close{box-sizing:content-box;width:1em;height:1em;padding:.25em .25em;color:#000;background:transparent 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 011.414 0L8 6.586 14.293.293a1 1 0 111.414 1.414L9.414 8l6.293 6.293a1 1 0 01-1.414 1.414L8 9.414l-6.293 6.293a1 1 0 01-1.414-1.414L6.586 8 .293 1.707a1 1 0 010-1.414z'/%3e%3c/svg%3e") center/1em auto no-repeat;border:0;border-radius:.25rem;opacity:.5}.btn-close:hover{color:#000;text-decoration:none;opacity:.75}.btn-close:focus{outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25);opacity:1}.btn-close.disabled,.btn-close:disabled{pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none;opacity:.25}.btn-close-white{filter:invert(1) grayscale(100%) brightness(200%)}.toast{width:350px;max-width:100%;font-size:.875rem;pointer-events:auto;background-color:rgba(255,255,255,.85);background-clip:padding-box;border:1px solid rgba(0,0,0,.1);box-shadow:0 .5rem 1rem rgba(0,0,0,.15);border-radius:.25rem}.toast:not(.showing):not(.show){opacity:0}.toast.hide{display:none}.toast-container{width:-webkit-max-content;width:-moz-max-content;width:max-content;max-width:100%;pointer-events:none}.toast-container>:not(:last-child){margin-bottom:.75rem}.toast-header{display:flex;align-items:center;padding:.5rem .75rem;color:#6c757d;background-color:rgba(255,255,255,.85);background-clip:padding-box;border-bottom:1px solid rgba(0,0,0,.05);border-top-left-radius:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.toast-header .btn-close{margin-right:-.375rem;margin-left:.75rem}.toast-body{padding:.75rem;word-wrap:break-word}.modal{position:fixed;top:0;left:0;z-index:1060;display:none;width:100%;height:100%;overflow-x:hidden;overflow-y:auto;outline:0}.modal-dialog{position:relative;width:auto;margin:.5rem;pointer-events:none}.modal.fade .modal-dialog{transition:transform .3s ease-out;transform:translate(0,-50px)}@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% - 1rem)}.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% - 1rem)}.modal-content{position:relative;display:flex;flex-direction:column;width:100%;pointer-events:auto;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem;outline:0}.modal-backdrop{position:fixed;top:0;left:0;z-index:1040;width:100vw;height:100vh;background-color:#000}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:.5}.modal-header{display:flex;flex-shrink:0;align-items:center;justify-content:space-between;padding:1rem 1rem;border-bottom:1px solid #dee2e6;border-top-left-radius:calc(.3rem - 1px);border-top-right-radius:calc(.3rem - 1px)}.modal-header .btn-close{padding:.5rem .5rem;margin:-.5rem -.5rem -.5rem auto}.modal-title{margin-bottom:0;line-height:1.5}.modal-body{position:relative;flex:1 1 auto;padding:1rem}.modal-footer{display:flex;flex-wrap:wrap;flex-shrink:0;align-items:center;justify-content:flex-end;padding:.75rem;border-top:1px solid #dee2e6;border-bottom-right-radius:calc(.3rem - 1px);border-bottom-left-radius:calc(.3rem - 1px)}.modal-footer>*{margin:.25rem}@media (min-width:576px){.modal-dialog{max-width:500px;margin:1.75rem auto}.modal-dialog-scrollable{height:calc(100% - 3.5rem)}.modal-dialog-centered{min-height:calc(100% - 3.5rem)}.modal-sm{max-width:300px}}@media (min-width:992px){.modal-lg,.modal-xl{max-width:800px}}@media (min-width:1200px){.modal-xl{max-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-header{border-radius:0}.modal-fullscreen .modal-body{overflow-y:auto}.modal-fullscreen .modal-footer{border-radius:0}@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-header{border-radius:0}.modal-fullscreen-sm-down .modal-body{overflow-y:auto}.modal-fullscreen-sm-down .modal-footer{border-radius:0}}@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-header{border-radius:0}.modal-fullscreen-md-down .modal-body{overflow-y:auto}.modal-fullscreen-md-down .modal-footer{border-radius:0}}@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-header{border-radius:0}.modal-fullscreen-lg-down .modal-body{overflow-y:auto}.modal-fullscreen-lg-down .modal-footer{border-radius:0}}@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-header{border-radius:0}.modal-fullscreen-xl-down .modal-body{overflow-y:auto}.modal-fullscreen-xl-down .modal-footer{border-radius:0}}@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-header{border-radius:0}.modal-fullscreen-xxl-down .modal-body{overflow-y:auto}.modal-fullscreen-xxl-down .modal-footer{border-radius:0}}.tooltip{position:absolute;z-index:1080;display:block;margin:0;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;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;opacity:0}.tooltip.show{opacity:.9}.tooltip .tooltip-arrow{position:absolute;display:block;width:.8rem;height:.4rem}.tooltip .tooltip-arrow::before{position:absolute;content:"";border-color:transparent;border-style:solid}.bs-tooltip-auto[data-popper-placement^=top],.bs-tooltip-top{padding:.4rem 0}.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow,.bs-tooltip-top .tooltip-arrow{bottom:0}.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow::before,.bs-tooltip-top .tooltip-arrow::before{top:-1px;border-width:.4rem .4rem 0;border-top-color:#000}.bs-tooltip-auto[data-popper-placement^=right],.bs-tooltip-end{padding:0 .4rem}.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow,.bs-tooltip-end .tooltip-arrow{left:0;width:.4rem;height:.8rem}.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow::before,.bs-tooltip-end .tooltip-arrow::before{right:-1px;border-width:.4rem .4rem .4rem 0;border-right-color:#000}.bs-tooltip-auto[data-popper-placement^=bottom],.bs-tooltip-bottom{padding:.4rem 0}.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow,.bs-tooltip-bottom .tooltip-arrow{top:0}.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow::before,.bs-tooltip-bottom .tooltip-arrow::before{bottom:-1px;border-width:0 .4rem .4rem;border-bottom-color:#000}.bs-tooltip-auto[data-popper-placement^=left],.bs-tooltip-start{padding:0 .4rem}.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow,.bs-tooltip-start .tooltip-arrow{right:0;width:.4rem;height:.8rem}.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow::before,.bs-tooltip-start .tooltip-arrow::before{left:-1px;border-width:.4rem 0 .4rem .4rem;border-left-color:#000}.tooltip-inner{max-width:200px;padding:.25rem .5rem;color:#fff;text-align:center;background-color:#000;border-radius:.25rem}.popover{position:absolute;top:0;left:0;z-index:1070;display:block;max-width:276px;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;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem}.popover .popover-arrow{position:absolute;display:block;width:1rem;height:.5rem}.popover .popover-arrow::after,.popover .popover-arrow::before{position:absolute;display:block;content:"";border-color:transparent;border-style:solid}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow,.bs-popover-top>.popover-arrow{bottom:calc(-.5rem - 1px)}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::before,.bs-popover-top>.popover-arrow::before{bottom:0;border-width:.5rem .5rem 0;border-top-color:rgba(0,0,0,.25)}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::after,.bs-popover-top>.popover-arrow::after{bottom:1px;border-width:.5rem .5rem 0;border-top-color:#fff}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow,.bs-popover-end>.popover-arrow{left:calc(-.5rem - 1px);width:.5rem;height:1rem}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::before,.bs-popover-end>.popover-arrow::before{left:0;border-width:.5rem .5rem .5rem 0;border-right-color:rgba(0,0,0,.25)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::after,.bs-popover-end>.popover-arrow::after{left:1px;border-width:.5rem .5rem .5rem 0;border-right-color:#fff}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow,.bs-popover-bottom>.popover-arrow{top:calc(-.5rem - 1px)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::before,.bs-popover-bottom>.popover-arrow::before{top:0;border-width:0 .5rem .5rem .5rem;border-bottom-color:rgba(0,0,0,.25)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::after,.bs-popover-bottom>.popover-arrow::after{top:1px;border-width:0 .5rem .5rem .5rem;border-bottom-color:#fff}.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:1rem;margin-left:-.5rem;content:"";border-bottom:1px solid #f0f0f0}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow,.bs-popover-start>.popover-arrow{right:calc(-.5rem - 1px);width:.5rem;height:1rem}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::before,.bs-popover-start>.popover-arrow::before{right:0;border-width:.5rem 0 .5rem .5rem;border-left-color:rgba(0,0,0,.25)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::after,.bs-popover-start>.popover-arrow::after{right:1px;border-width:.5rem 0 .5rem .5rem;border-left-color:#fff}.popover-header{padding:.5rem 1rem;margin-bottom:0;font-size:1rem;background-color:#f0f0f0;border-bottom:1px solid rgba(0,0,0,.2);border-top-left-radius:calc(.3rem - 1px);border-top-right-radius:calc(.3rem - 1px)}.popover-header:empty{display:none}.popover-body{padding:1rem 1rem;color:#212529}.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;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 0z'/%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-.708z'/%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%;list-style:none}.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:#fff;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:#fff;text-align:center}.carousel-dark .carousel-control-next-icon,.carousel-dark .carousel-control-prev-icon{filter:invert(1) grayscale(100)}.carousel-dark .carousel-indicators [data-bs-target]{background-color:#000}.carousel-dark .carousel-caption{color:#000}@-webkit-keyframes spinner-border{to{transform:rotate(360deg)}}@keyframes spinner-border{to{transform:rotate(360deg)}}.spinner-border{display:inline-block;width:2rem;height:2rem;vertical-align:-.125em;border:.25em solid currentColor;border-right-color:transparent;border-radius:50%;-webkit-animation:.75s linear infinite spinner-border;animation:.75s linear infinite spinner-border}.spinner-border-sm{width:1rem;height:1rem;border-width:.2em}@-webkit-keyframes spinner-grow{0%{transform:scale(0)}50%{opacity:1;transform:none}}@keyframes spinner-grow{0%{transform:scale(0)}50%{opacity:1;transform:none}}.spinner-grow{display:inline-block;width:2rem;height:2rem;vertical-align:-.125em;background-color:currentColor;border-radius:50%;opacity:0;-webkit-animation:.75s linear infinite spinner-grow;animation:.75s linear infinite spinner-grow}.spinner-grow-sm{width:1rem;height:1rem}@media (prefers-reduced-motion:reduce){.spinner-border,.spinner-grow{-webkit-animation-duration:1.5s;animation-duration:1.5s}}.offcanvas{position:fixed;bottom:0;z-index:1050;display:flex;flex-direction:column;max-width:100%;visibility:hidden;background-color:#fff;background-clip:padding-box;outline:0;transition:transform .3s ease-in-out}@media (prefers-reduced-motion:reduce){.offcanvas{transition:none}}.offcanvas-header{display:flex;align-items:center;justify-content:space-between;padding:1rem 1rem}.offcanvas-header .btn-close{padding:.5rem .5rem;margin-top:-.5rem;margin-right:-.5rem;margin-bottom:-.5rem}.offcanvas-title{margin-bottom:0;line-height:1.5}.offcanvas-body{flex-grow:1;padding:1rem 1rem;overflow-y:auto}.offcanvas-start{top:0;left:0;width:400px;border-right:1px solid rgba(0,0,0,.2);transform:translateX(-100%)}.offcanvas-end{top:0;right:0;width:400px;border-left:1px solid rgba(0,0,0,.2);transform:translateX(100%)}.offcanvas-top{top:0;right:0;left:0;height:30vh;max-height:100%;border-bottom:1px solid rgba(0,0,0,.2);transform:translateY(-100%)}.offcanvas-bottom{right:0;left:0;height:30vh;max-height:100%;border-top:1px solid rgba(0,0,0,.2);transform:translateY(100%)}.offcanvas.show{transform:none}.clearfix::after{display:block;clear:both;content:""}.link-primary{color:#0d6efd}.link-primary:focus,.link-primary:hover{color:#0a58ca}.link-secondary{color:#6c757d}.link-secondary:focus,.link-secondary:hover{color:#565e64}.link-success{color:#198754}.link-success:focus,.link-success:hover{color:#146c43}.link-info{color:#0dcaf0}.link-info:focus,.link-info:hover{color:#3dd5f3}.link-warning{color:#ffc107}.link-warning:focus,.link-warning:hover{color:#ffcd39}.link-danger{color:#dc3545}.link-danger:focus,.link-danger:hover{color:#b02a37}.link-light{color:#f8f9fa}.link-light:focus,.link-light:hover{color:#f9fafb}.link-dark{color:#212529}.link-dark:focus,.link-dark:hover{color:#1a1e21}.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:calc(3 / 4 * 100%)}.ratio-16x9{--bs-aspect-ratio:calc(9 / 16 * 100%)}.ratio-21x9{--bs-aspect-ratio:calc(9 / 21 * 100%)}.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}@media (min-width:576px){.sticky-sm-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}@media (min-width:768px){.sticky-md-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}@media (min-width:992px){.sticky-lg-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}@media (min-width:1200px){.sticky-xl-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}@media (min-width:1400px){.sticky-xxl-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}.visually-hidden,.visually-hidden-focusable:not(:focus):not(:focus-within){position:absolute!important;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}.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}.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}.overflow-auto{overflow:auto!important}.overflow-hidden{overflow:hidden!important}.overflow-visible{overflow:visible!important}.overflow-scroll{overflow: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-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:0 .5rem 1rem rgba(0,0,0,.15)!important}.shadow-sm{box-shadow:0 .125rem .25rem rgba(0,0,0,.075)!important}.shadow-lg{box-shadow:0 1rem 3rem rgba(0,0,0,.175)!important}.shadow-none{box-shadow:none!important}.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:1px solid #dee2e6!important}.border-0{border:0!important}.border-top{border-top:1px solid #dee2e6!important}.border-top-0{border-top:0!important}.border-end{border-right:1px solid #dee2e6!important}.border-end-0{border-right:0!important}.border-bottom{border-bottom:1px solid #dee2e6!important}.border-bottom-0{border-bottom:0!important}.border-start{border-left:1px solid #dee2e6!important}.border-start-0{border-left:0!important}.border-primary{border-color:#0d6efd!important}.border-secondary{border-color:#6c757d!important}.border-success{border-color:#198754!important}.border-info{border-color:#0dcaf0!important}.border-warning{border-color:#ffc107!important}.border-danger{border-color:#dc3545!important}.border-light{border-color:#f8f9fa!important}.border-dark{border-color:#212529!important}.border-white{border-color:#fff!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}.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}.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}.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}.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-light{font-weight:300!important}.fw-lighter{font-weight:lighter!important}.fw-normal{font-weight:400!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{color:#0d6efd!important}.text-secondary{color:#6c757d!important}.text-success{color:#198754!important}.text-info{color:#0dcaf0!important}.text-warning{color:#ffc107!important}.text-danger{color:#dc3545!important}.text-light{color:#f8f9fa!important}.text-dark{color:#212529!important}.text-white{color:#fff!important}.text-body{color:#212529!important}.text-muted{color:#6c757d!important}.text-black-50{color:rgba(0,0,0,.5)!important}.text-white-50{color:rgba(255,255,255,.5)!important}.text-reset{color:inherit!important}.bg-primary{background-color:#0d6efd!important}.bg-secondary{background-color:#6c757d!important}.bg-success{background-color:#198754!important}.bg-info{background-color:#0dcaf0!important}.bg-warning{background-color:#ffc107!important}.bg-danger{background-color:#dc3545!important}.bg-light{background-color:#f8f9fa!important}.bg-dark{background-color:#212529!important}.bg-body{background-color:#fff!important}.bg-white{background-color:#fff!important}.bg-transparent{background-color:transparent!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:.25rem!important}.rounded-0{border-radius:0!important}.rounded-1{border-radius:.2rem!important}.rounded-2{border-radius:.25rem!important}.rounded-3{border-radius:.3rem!important}.rounded-circle{border-radius:50%!important}.rounded-pill{border-radius:50rem!important}.rounded-top{border-top-left-radius:.25rem!important;border-top-right-radius:.25rem!important}.rounded-end{border-top-right-radius:.25rem!important;border-bottom-right-radius:.25rem!important}.rounded-bottom{border-bottom-right-radius:.25rem!important;border-bottom-left-radius:.25rem!important}.rounded-start{border-bottom-left-radius:.25rem!important;border-top-left-radius:.25rem!important}.visible{visibility:visible!important}.invisible{visibility:hidden!important}@media (min-width:576px){.float-sm-start{float:left!important}.float-sm-end{float:right!important}.float-sm-none{float: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-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}.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}.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}.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}.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-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}.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}.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}.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}.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-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}.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}.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}.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}.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-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}.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}.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}.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}.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-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}.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}.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}.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-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}} +/*# sourceMappingURL=bootstrap.min.css.map */ \ No newline at end of file diff --git a/src/main/resources/templates/Post/post.html b/src/main/resources/templates/Post/post.html new file mode 100644 index 0000000..6c9d7e4 --- /dev/null +++ b/src/main/resources/templates/Post/post.html @@ -0,0 +1,324 @@ + + + + + + 레시피 등록 + + + + + + + +
+
+
+ + +
+
+

레시피 등록

+ +
+ +
+ + +
+ + +
+ +
+ + +
+
+ +
+ + +

재료 정보

+

재료가 남거나 부족하지 않도록 정확한 계량정보를 적어주세요.

+ +
+
+
+ +
+
+ +
+
+ +
+
+ +
+
+
+ + + +
+ + +

요리 순서

+

+ 요리의 맛이 좌우될 수 있는 중요한 부분은 빠짐없이 적어주세요.
+ 예) 10분간 약한불로 익혀주세요. 마늘편은 충분히 익혀 매운 맛을 제거하세요. +

+ +
+
+ +
+ + +
+
+
+ + + +
+ +
+
+ +
+
+
+ + + + + From 9f9e83a6e56444b82c0a45561b4454bba83457a1 Mon Sep 17 00:00:00 2001 From: phc979 Date: Mon, 10 Feb 2025 09:26:39 +0900 Subject: [PATCH 05/73] feature : DTOs Annotation fix --- src/main/java/com/recipe/cookofking/dto/LikeDto.java | 7 ++++--- src/main/java/com/recipe/cookofking/dto/UserDto.java | 7 ++++--- .../recipe/cookofking/dto/image/ImageValidationDto.java | 6 ++++++ .../com/recipe/cookofking/dto/image/ImagemappingDto.java | 7 ++++--- src/main/java/com/recipe/cookofking/dto/post/PostDto.java | 7 ++++--- .../recipe/cookofking/dto/post/RecipeSubmissionDto.java | 6 ++++++ 6 files changed, 28 insertions(+), 12 deletions(-) diff --git a/src/main/java/com/recipe/cookofking/dto/LikeDto.java b/src/main/java/com/recipe/cookofking/dto/LikeDto.java index 871c24a..bc0382f 100644 --- a/src/main/java/com/recipe/cookofking/dto/LikeDto.java +++ b/src/main/java/com/recipe/cookofking/dto/LikeDto.java @@ -2,8 +2,7 @@ import com.recipe.cookofking.dto.post.PostDto; import com.recipe.cookofking.entity.Like; -import lombok.Builder; -import lombok.Value; +import lombok.*; import java.io.Serializable; import java.time.LocalDateTime; @@ -11,8 +10,10 @@ /** * DTO for {@link Like} */ +@Data +@AllArgsConstructor +@NoArgsConstructor @Builder -@Value public class LikeDto implements Serializable { Integer id; PostDto post; diff --git a/src/main/java/com/recipe/cookofking/dto/UserDto.java b/src/main/java/com/recipe/cookofking/dto/UserDto.java index 88c0e85..14e9bc3 100644 --- a/src/main/java/com/recipe/cookofking/dto/UserDto.java +++ b/src/main/java/com/recipe/cookofking/dto/UserDto.java @@ -1,8 +1,7 @@ package com.recipe.cookofking.dto; import com.recipe.cookofking.entity.User; -import lombok.Builder; -import lombok.Value; +import lombok.*; import java.io.Serializable; import java.time.LocalDateTime; @@ -10,8 +9,10 @@ /** * DTO for {@link User} */ +@Data +@AllArgsConstructor +@NoArgsConstructor @Builder -@Value public class UserDto implements Serializable { Integer id; String username; diff --git a/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java b/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java index 3e25b31..12732d0 100644 --- a/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java +++ b/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java @@ -1,10 +1,16 @@ package com.recipe.cookofking.dto.image; +import lombok.AllArgsConstructor; +import lombok.Builder; import lombok.Data; +import lombok.NoArgsConstructor; import java.util.List; @Data +@AllArgsConstructor +@NoArgsConstructor +@Builder public class ImageValidationDto { private Integer mainImageId; private String mainImageUrl; diff --git a/src/main/java/com/recipe/cookofking/dto/image/ImagemappingDto.java b/src/main/java/com/recipe/cookofking/dto/image/ImagemappingDto.java index ac9296b..2b702d2 100644 --- a/src/main/java/com/recipe/cookofking/dto/image/ImagemappingDto.java +++ b/src/main/java/com/recipe/cookofking/dto/image/ImagemappingDto.java @@ -2,8 +2,7 @@ import com.recipe.cookofking.dto.post.PostDto; import com.recipe.cookofking.entity.Imagemapping; -import lombok.Builder; -import lombok.Value; +import lombok.*; import java.io.Serializable; import java.time.LocalDateTime; @@ -11,8 +10,10 @@ /** * DTO for {@link Imagemapping} */ +@Data +@AllArgsConstructor +@NoArgsConstructor @Builder -@Value public class ImagemappingDto implements Serializable { Integer id; PostDto post; diff --git a/src/main/java/com/recipe/cookofking/dto/post/PostDto.java b/src/main/java/com/recipe/cookofking/dto/post/PostDto.java index fc8d9df..bdab910 100644 --- a/src/main/java/com/recipe/cookofking/dto/post/PostDto.java +++ b/src/main/java/com/recipe/cookofking/dto/post/PostDto.java @@ -2,8 +2,7 @@ import com.recipe.cookofking.dto.UserDto; import com.recipe.cookofking.entity.Post; -import lombok.Builder; -import lombok.Value; +import lombok.*; import java.io.Serializable; import java.time.LocalDateTime; @@ -11,8 +10,10 @@ /** * DTO for {@link Post} */ +@Data +@AllArgsConstructor +@NoArgsConstructor @Builder -@Value public class PostDto implements Serializable { Integer id; UserDto user; diff --git a/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java b/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java index fdab3f9..6289c2d 100644 --- a/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java +++ b/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java @@ -1,9 +1,15 @@ package com.recipe.cookofking.dto.post; import com.recipe.cookofking.dto.image.ImageValidationDto; +import lombok.AllArgsConstructor; +import lombok.Builder; import lombok.Data; +import lombok.NoArgsConstructor; @Data +@AllArgsConstructor +@NoArgsConstructor +@Builder public class RecipeSubmissionDto { private ImageValidationDto validationData; private PostDto recipeData; From 636c645b468fc5ac8ffa397480a283fca648b1e0 Mon Sep 17 00:00:00 2001 From: KangSeongKwan Date: Mon, 10 Feb 2025 13:33:03 +0900 Subject: [PATCH 06/73] Make View Like but Returned 500 Error Version --- .../cookofking/controller/LikeController.java | 22 ++++++++++++++++++- .../com/recipe/cookofking/entity/Like.java | 2 -- .../cookofking/repository/LikeRepository.java | 1 + .../cookofking/repository/PostRepository.java | 2 ++ .../cookofking/service/LikeService.java | 21 ++++++++++++++++++ 5 files changed, 45 insertions(+), 3 deletions(-) diff --git a/src/main/java/com/recipe/cookofking/controller/LikeController.java b/src/main/java/com/recipe/cookofking/controller/LikeController.java index e1844e9..08dc0e3 100644 --- a/src/main/java/com/recipe/cookofking/controller/LikeController.java +++ b/src/main/java/com/recipe/cookofking/controller/LikeController.java @@ -1,10 +1,30 @@ package com.recipe.cookofking.controller; +import com.recipe.cookofking.dto.LikeDto; +import com.recipe.cookofking.service.LikeService; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestParam; import org.springframework.web.bind.annotation.RestController; + @RestController -@RequestMapping("/likes") +@RequestMapping("/api/v1/likes") public class LikeController { + private final LikeService likeService; + + public LikeController(LikeService likeService) { + this.likeService = likeService; + } + @GetMapping + public ResponseEntity getLikes(@RequestParam Integer postId) { + try { + LikeDto responseDto = likeService.getLikes(postId); + return ResponseEntity.ok(responseDto); + } catch (IllegalArgumentException e) { + return ResponseEntity.badRequest().body(e.getMessage()); + } + } } diff --git a/src/main/java/com/recipe/cookofking/entity/Like.java b/src/main/java/com/recipe/cookofking/entity/Like.java index 0c23e89..bf77710 100644 --- a/src/main/java/com/recipe/cookofking/entity/Like.java +++ b/src/main/java/com/recipe/cookofking/entity/Like.java @@ -31,6 +31,4 @@ public class Like { @Column(name = "createdDate") @CreatedDate private LocalDateTime createdDate; - - } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/repository/LikeRepository.java b/src/main/java/com/recipe/cookofking/repository/LikeRepository.java index 32a3b9f..749c434 100644 --- a/src/main/java/com/recipe/cookofking/repository/LikeRepository.java +++ b/src/main/java/com/recipe/cookofking/repository/LikeRepository.java @@ -6,4 +6,5 @@ @Repository public interface LikeRepository extends JpaRepository { // Add custom query methods if needed + long countByPostId(Integer postId); // 특정 게시물의 좋아요 개수를 반환 } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/repository/PostRepository.java b/src/main/java/com/recipe/cookofking/repository/PostRepository.java index 6469422..4a24eb8 100644 --- a/src/main/java/com/recipe/cookofking/repository/PostRepository.java +++ b/src/main/java/com/recipe/cookofking/repository/PostRepository.java @@ -1,7 +1,9 @@ package com.recipe.cookofking.repository; + import com.recipe.cookofking.entity.Post; import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.stereotype.Repository; + @Repository public interface PostRepository extends JpaRepository { // Add custom query methods if needed diff --git a/src/main/java/com/recipe/cookofking/service/LikeService.java b/src/main/java/com/recipe/cookofking/service/LikeService.java index adb277e..56d8cdc 100644 --- a/src/main/java/com/recipe/cookofking/service/LikeService.java +++ b/src/main/java/com/recipe/cookofking/service/LikeService.java @@ -1,6 +1,27 @@ package com.recipe.cookofking.service; + +import com.recipe.cookofking.dto.LikeDto; +import com.recipe.cookofking.repository.LikeRepository; +import com.recipe.cookofking.repository.PostRepository; import org.springframework.stereotype.Service; + @Service public class LikeService { // Add Like related business logic here + private final LikeRepository likeRepository; + private final PostRepository postRepository; + + public LikeService(LikeRepository likeRepository, PostRepository postRepository) { + this.likeRepository = likeRepository; + this.postRepository = postRepository; + } + + public LikeDto getLikes(Integer postId) { + if (!postRepository.existsById(postId)) { + throw new IllegalArgumentException("Post not found"); + } + + long likesCount = likeRepository.countByPostId(postId); + return new LikeDto(); + } } From bf8d4ed144ee5adb115741f57f7c2de6acf97371 Mon Sep 17 00:00:00 2001 From: phc979 Date: Mon, 10 Feb 2025 13:40:56 +0900 Subject: [PATCH 07/73] feature : post-viewpage & fix url mapping --- .../controller/ImagemappingController.java | 3 +- .../cookofking/controller/PostController.java | 26 ++- .../controller/PostRestController.java | 2 +- .../com/recipe/cookofking/entity/Post.java | 4 +- .../recipe/cookofking/mapper/PostMapper.java | 2 + .../service/ImagemappingService.java | 3 +- .../cookofking/service/PostService.java | 9 + .../resources/templates/Post/post-view.html | 192 ++++++++++++++++++ .../Post/{post.html => post-write.html} | 6 +- 9 files changed, 236 insertions(+), 11 deletions(-) create mode 100644 src/main/resources/templates/Post/post-view.html rename src/main/resources/templates/Post/{post.html => post-write.html} (99%) diff --git a/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java b/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java index e3c2605..50073fb 100644 --- a/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java +++ b/src/main/java/com/recipe/cookofking/controller/ImagemappingController.java @@ -11,13 +11,14 @@ import org.springframework.web.multipart.MultipartFile; @RestController -@RequestMapping("/upload-image") +@RequestMapping("/api/v1") @RequiredArgsConstructor public class ImagemappingController { private final ImagemappingService imagemappingService; @PostMapping + @RequestMapping("/upload-image") public ResponseEntity uploadImage(@RequestParam("file") MultipartFile file) { try { ImagemappingDto uploadedImage = imagemappingService.uploadImage(file); diff --git a/src/main/java/com/recipe/cookofking/controller/PostController.java b/src/main/java/com/recipe/cookofking/controller/PostController.java index e239f71..0cc654b 100644 --- a/src/main/java/com/recipe/cookofking/controller/PostController.java +++ b/src/main/java/com/recipe/cookofking/controller/PostController.java @@ -1,13 +1,33 @@ package com.recipe.cookofking.controller; +import com.recipe.cookofking.dto.post.PostDto; +import com.recipe.cookofking.service.PostService; +import lombok.RequiredArgsConstructor; import org.springframework.stereotype.Controller; +import org.springframework.ui.Model; +import org.springframework.web.bind.annotation.PathVariable; import org.springframework.web.bind.annotation.RequestMapping; @Controller +@RequiredArgsConstructor +@RequestMapping("/post") public class PostController { - @RequestMapping("/") - public String showForm() { - return "Post/post"; // 레시피 작성 폼 렌더링 + private final PostService postService; + + // 레시피 작성 폼 + @RequestMapping("/write") + public String showWriteForm() { + return "Post/post-write"; } + + + // 레시피 조회 폼 + @RequestMapping("/view/{postid}") + public String showViewForm(@PathVariable Integer postid, Model model) { + PostDto postDto = postService.getPostById(postid); // postid로 게시글 데이터 가져오기 + model.addAttribute("post", postDto); // 모델에 데이터 추가 + return "Post/post-view"; // post-view.html 렌더링 + } + } diff --git a/src/main/java/com/recipe/cookofking/controller/PostRestController.java b/src/main/java/com/recipe/cookofking/controller/PostRestController.java index 2e1266a..6e73c1d 100644 --- a/src/main/java/com/recipe/cookofking/controller/PostRestController.java +++ b/src/main/java/com/recipe/cookofking/controller/PostRestController.java @@ -8,7 +8,7 @@ import org.springframework.web.bind.annotation.*; @RestController -@RequestMapping("/api") +@RequestMapping("/api/v1") @RequiredArgsConstructor public class PostRestController { diff --git a/src/main/java/com/recipe/cookofking/entity/Post.java b/src/main/java/com/recipe/cookofking/entity/Post.java index 044c240..2a4daaa 100644 --- a/src/main/java/com/recipe/cookofking/entity/Post.java +++ b/src/main/java/com/recipe/cookofking/entity/Post.java @@ -30,11 +30,11 @@ public class Post { private String title; @Lob - @Column(name = "content", nullable = false) + @Column(name = "content", nullable = false, columnDefinition = "TEXT") private String content; @Lob - @Column(name = "ingredients") + @Column(name = "ingredients", columnDefinition = "TEXT") private String ingredients; @Lob diff --git a/src/main/java/com/recipe/cookofking/mapper/PostMapper.java b/src/main/java/com/recipe/cookofking/mapper/PostMapper.java index c6d4bc8..4222da6 100644 --- a/src/main/java/com/recipe/cookofking/mapper/PostMapper.java +++ b/src/main/java/com/recipe/cookofking/mapper/PostMapper.java @@ -15,6 +15,7 @@ public static PostDto toDto(Post post) { .content(post.getContent()) .ingredients(post.getIngredients()) .instructions(post.getInstructions()) + .mainImageS3URL(post.getMainImageS3URL()) .createdDate(post.getCreatedDate()) .modifiedDate(post.getModifiedDate()) .build(); @@ -31,6 +32,7 @@ public static Post toEntity(PostDto postDto) { .content(postDto.getContent()) .ingredients(postDto.getIngredients()) .instructions(postDto.getInstructions()) + .mainImageS3URL(postDto.getMainImageS3URL()) .createdDate(postDto.getCreatedDate()) .modifiedDate(postDto.getModifiedDate()) .build(); diff --git a/src/main/java/com/recipe/cookofking/service/ImagemappingService.java b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java index b4163bf..db36363 100644 --- a/src/main/java/com/recipe/cookofking/service/ImagemappingService.java +++ b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java @@ -40,7 +40,8 @@ public ImagemappingDto uploadImage(MultipartFile file) throws IOException { s3Client.putObject(putObjectRequest, software.amazon.awssdk.core.sync.RequestBody.fromInputStream(file.getInputStream(), file.getSize())); // 업로드된 파일의 S3 URL 생성 - String s3Url = "https://" + bucketName + ".s3.amazonaws.com/" + fileName; + String region = s3Client.serviceClientConfiguration().region().id(); + String s3Url = "https://" + bucketName + ".s3." + region + ".amazonaws.com/" + fileName; // DB에 이미지 정보 저장 Imagemapping imagemapping = Imagemapping.builder() diff --git a/src/main/java/com/recipe/cookofking/service/PostService.java b/src/main/java/com/recipe/cookofking/service/PostService.java index 8820ae5..d7ae7b2 100644 --- a/src/main/java/com/recipe/cookofking/service/PostService.java +++ b/src/main/java/com/recipe/cookofking/service/PostService.java @@ -1,6 +1,7 @@ package com.recipe.cookofking.service; import com.recipe.cookofking.dto.post.PostDto; import com.recipe.cookofking.entity.Post; +import com.recipe.cookofking.mapper.PostMapper; import com.recipe.cookofking.repository.PostRepository; import lombok.RequiredArgsConstructor; import org.springframework.stereotype.Service; @@ -25,4 +26,12 @@ public void savePost(PostDto postDto) { // 2. 레시피 저장 Post savedPost = postRepository.save(post); } + + @Transactional(readOnly = true) + public PostDto getPostById(Integer id) { + Post post = postRepository.findById(id) + .orElseThrow(() -> new RuntimeException("Post not found with id: " + id)); + return PostMapper.toDto(post); // 엔티티를 DTO로 변환 + } + } diff --git a/src/main/resources/templates/Post/post-view.html b/src/main/resources/templates/Post/post-view.html new file mode 100644 index 0000000..b982e9c --- /dev/null +++ b/src/main/resources/templates/Post/post-view.html @@ -0,0 +1,192 @@ + + + + + + 레시피 상세 보기 + + + + + + + +
+
+ + +
+
+
+ + +
+
+ + +
+ 메인 이미지 +
+ + +

레시피 제목

+ + +

요리 소개

+ + +
+ 작성자: 작성자 이름 + 조회수: 3740 +
+ + +
+ + + +

+ 좋아요 수: 0 +

+
+ +
+
+
+
+
+ +
+
+
+ +

재료 정보

+
    +
    +
    + +
    +
    + +

    조리 순서

    +
    + + 목록으로 돌아가기 +
    +
    +
    + +
    +
    + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/templates/Post/post.html b/src/main/resources/templates/Post/post-write.html similarity index 99% rename from src/main/resources/templates/Post/post.html rename to src/main/resources/templates/Post/post-write.html index 6c9d7e4..963d67e 100644 --- a/src/main/resources/templates/Post/post.html +++ b/src/main/resources/templates/Post/post-write.html @@ -157,7 +157,7 @@

    요리 순서

    formData.append('file', file); $.ajax({ - url: '/upload-image', + url: '/api/v1/upload-image', type: 'POST', data: formData, processData: false, @@ -183,7 +183,7 @@

    요리 순서

    formData.append('file', file); $.ajax({ - url: '/upload-image', + url: '/api/v1/upload-image', type: 'POST', data: formData, processData: false, @@ -226,7 +226,7 @@

    요리 순서

    // JSON 데이터 전송 $.ajax({ - url: '/api/submit-recipe', + url: '/api/v1/submit-recipe', type: 'POST', contentType: 'application/json', data: JSON.stringify(finalData), From 6f82a32bd33e15f9916ed43a3696f654464cf287 Mon Sep 17 00:00:00 2001 From: SUJINJEONG012 Date: Mon, 10 Feb 2025 14:06:14 +0900 Subject: [PATCH 08/73] =?UTF-8?q?=ED=98=84=EC=9E=AC=20=ED=9A=8C=EC=9B=90?= =?UTF-8?q?=EA=B0=80=EC=9E=85=EA=B9=8C=EC=A7=80=20=EC=A0=80=EC=9E=A5,jwt?= =?UTF-8?q?=EC=9C=BC=EB=A1=9C=20=EB=A1=9C=EA=B7=B8=EC=9D=B8=EA=B5=AC?= =?UTF-8?q?=ED=98=84=EC=A4=91?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- build.gradle | 2 + gradlew | 0 .../cookofking/config/SecurityConfig.java | 61 ++-------- .../config/auth/PrincipalDetails.java | 74 ++++++++++++ .../config/auth/PrincipalDetailsService.java | 32 +++++ .../config/jwt/JwtAuthenticationFilter.java | 90 ++++++++++++++ .../config/jwt/JwtAuthorizationFilter.java | 78 ++++++++++++ .../cookofking/config/jwt/JwtProperties.java | 9 ++ .../cookofking/controller/PostController.java | 3 + .../cookofking/controller/UserController.java | 113 +++++++++++++++++- .../com/recipe/cookofking/dto/UserDto.java | 47 +++++++- .../cookofking/dto/request/UserRequest.java | 37 ++++++ .../com/recipe/cookofking/entity/User.java | 10 ++ .../cookofking/repository/UserRepository.java | 10 +- .../cookofking/service/UserService.java | 70 ++++++++++- src/main/resources/static/css/register.css | 103 ++++++++++++++++ src/main/resources/templates/user/login.html | 22 ++++ .../resources/templates/user/register.html | 62 ++++++++++ 18 files changed, 763 insertions(+), 60 deletions(-) mode change 100644 => 100755 gradlew create mode 100644 src/main/java/com/recipe/cookofking/config/auth/PrincipalDetails.java create mode 100644 src/main/java/com/recipe/cookofking/config/auth/PrincipalDetailsService.java create mode 100644 src/main/java/com/recipe/cookofking/config/jwt/JwtAuthenticationFilter.java create mode 100644 src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java create mode 100644 src/main/java/com/recipe/cookofking/config/jwt/JwtProperties.java create mode 100644 src/main/java/com/recipe/cookofking/dto/request/UserRequest.java create mode 100644 src/main/resources/static/css/register.css create mode 100644 src/main/resources/templates/user/login.html create mode 100644 src/main/resources/templates/user/register.html diff --git a/build.gradle b/build.gradle index 597ca3d..752ad81 100644 --- a/build.gradle +++ b/build.gradle @@ -30,6 +30,8 @@ dependencies { implementation 'org.springframework.boot:spring-boot-starter-thymeleaf' implementation 'org.springframework.boot:spring-boot-starter-web' implementation 'org.springframework.boot:spring-boot-starter-websocket' + implementation 'com.auth0:java-jwt:3.18.2' + implementation 'org.thymeleaf.extras:thymeleaf-extras-springsecurity6' implementation 'software.amazon.awssdk:s3:2.20.147' compileOnly 'org.projectlombok:lombok' diff --git a/gradlew b/gradlew old mode 100644 new mode 100755 diff --git a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java index 5a61be1..b101194 100644 --- a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java +++ b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java @@ -8,6 +8,8 @@ import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.security.config.annotation.web.builders.HttpSecurity; +import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; +import org.springframework.security.crypto.password.PasswordEncoder; import org.springframework.security.web.SecurityFilterChain; @Configuration @@ -24,56 +26,11 @@ public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Excepti return http.build(); } + + @Bean + public PasswordEncoder passwordEncoder() { + // BCryptPasswordEncoder를 사용해 비밀번호 암호화 + return new BCryptPasswordEncoder(); + } + } -// -// @Bean -// public SecurityFilterChain securityFilterChain(HttpSecurity security) throws Exception { -// -// security -// .csrf(csrf -> csrf.disable()); -// //.cors(cors -> cors.disable()); -// security.authorizeHttpRequests(authorizeRequests -> -// authorizeRequests -// .requestMatchers("/assets/**").permitAll() // Allow static assets -// .requestMatchers(PathRequest.toStaticResources().atCommonLocations()).permitAll() -// .requestMatchers("/u/details/**").hasRole("USER") -// .requestMatchers("/f/admin/**").hasRole("ADMIN") -// .requestMatchers("/login", "/register", "/", "/f/c", "/f", "/f/a", "/f/a/{artist_id}").permitAll() -// .anyRequest().authenticated() -// ); -// security.formLogin(form -> form -// .loginPage("/login") -// .loginProcessingUrl("/login") -// .usernameParameter("email") // 사용자 이름 필드를 이메일로 설정 -// .passwordParameter("password") // 비밀번호 필드 설정 -// .defaultSuccessUrl("/f/c")); //f/c -// -// security.logout(logout -> logout.logoutSuccessUrl("/f/c")); -// //세션 -// security.sessionManagement(httpSecuritySessionManagementConfigurer -> httpSecuritySessionManagementConfigurer -// .invalidSessionUrl("/login") -// .sessionFixation().migrateSession() //로그인후 세션 변경 -// .maximumSessions(1)//한개만 유지 -// .expiredUrl("/login"));//만료시 이동 -// -// return security.build(); -// -// } -// -// -// @Bean -// public UserDetailsService userDetailsService(UserService userService) { -// return email -> userService.searchUserByEmail(email).map(FanUserDetails::toFUDto) -// .orElseThrow(() -> new UsernameNotFoundException("User not found with email: " + email)); -// } -// -// -// -// @Bean -// public PasswordEncoder passwordEncoder() { -// -// return NoOpPasswordEncoder.getInstance(); -// return new BCryptPasswordEncoder(); -// } - -//} diff --git a/src/main/java/com/recipe/cookofking/config/auth/PrincipalDetails.java b/src/main/java/com/recipe/cookofking/config/auth/PrincipalDetails.java new file mode 100644 index 0000000..8fa1572 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/config/auth/PrincipalDetails.java @@ -0,0 +1,74 @@ +package com.recipe.cookofking.config.auth; + +import java.util.Arrays; +import java.util.Collection; + +import org.springframework.security.core.GrantedAuthority; +import org.springframework.security.core.authority.SimpleGrantedAuthority; +import org.springframework.security.core.userdetails.UserDetails; + +import com.recipe.cookofking.dto.UserDto; +import com.recipe.cookofking.entity.User; + +import lombok.Getter; +import lombok.RequiredArgsConstructor; + + +@Getter +@RequiredArgsConstructor +public class PrincipalDetails implements UserDetails { + + private static final long serialVersionUID = 1L; + + + private final User user; // User 엔티티를 직접 사용 + private final UserDto userDto; + + public PrincipalDetails(User user) { + this.user=user; + this.userDto=UserDto.fromEntity(user); + } + + + + @Override + public Collection getAuthorities() { + // user.getRoleType()이 null일 경우 기본값 설정 + if (user.getRole() == null) { + return Arrays.asList(new SimpleGrantedAuthority("ROLE_USER")); // 기본값으로 ROLE_USER 사용 + } + return Arrays.asList(new SimpleGrantedAuthority("ROLE_" + user.getRole())); + } + + @Override + public String getPassword() { + return user.getPassword(); // user 객체에서 password 가져오기 + } + + @Override + public String getUsername() { + return user.getUsername(); // user 객체에서 uid 가져오기 + } + + @Override + public boolean isAccountNonExpired() { + return true; + } + + @Override + public boolean isAccountNonLocked() { + return true; + } + + @Override + public boolean isCredentialsNonExpired() { + return true; + } + + @Override + public boolean isEnabled() { + return true; + } + + +} \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/config/auth/PrincipalDetailsService.java b/src/main/java/com/recipe/cookofking/config/auth/PrincipalDetailsService.java new file mode 100644 index 0000000..bff7ade --- /dev/null +++ b/src/main/java/com/recipe/cookofking/config/auth/PrincipalDetailsService.java @@ -0,0 +1,32 @@ +package com.recipe.cookofking.config.auth; + +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.security.core.userdetails.UserDetailsService; +import org.springframework.security.core.userdetails.UsernameNotFoundException; +import org.springframework.stereotype.Service; + +import com.recipe.cookofking.entity.User; +import com.recipe.cookofking.repository.UserRepository; + +import lombok.RequiredArgsConstructor; + +@RequiredArgsConstructor +@Service +public class PrincipalDetailsService implements UserDetailsService { + + private final UserRepository userRepository; + + @Override + public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException { + + // Optional에서 User를 안전하게 꺼내기 + User user = userRepository.findByUsername(username) + .orElseThrow(() -> new UsernameNotFoundException("User not found")); + + return new PrincipalDetails(user); + + + } + + +} diff --git a/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthenticationFilter.java b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthenticationFilter.java new file mode 100644 index 0000000..70867dc --- /dev/null +++ b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthenticationFilter.java @@ -0,0 +1,90 @@ +package com.recipe.cookofking.config.jwt; + + +import java.io.IOException; +import java.util.Date; + +import org.springframework.security.authentication.AuthenticationManager; +import org.springframework.security.authentication.UsernamePasswordAuthenticationToken; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.AuthenticationException; +import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter; + +import com.auth0.jwt.JWT; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.recipe.cookofking.config.auth.PrincipalDetails; +import com.recipe.cookofking.dto.UserDto; + +import jakarta.servlet.FilterChain; +import jakarta.servlet.ServletException; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpServletResponse; +import lombok.extern.slf4j.Slf4j; +import software.amazon.awssdk.core.checksums.Algorithm; + + +// 로그인처리 담당하는 필터 +@Slf4j +public class JwtAuthenticationFilter extends UsernamePasswordAuthenticationFilter { + + + private final AuthenticationManager authenticationManager; + + public JwtAuthenticationFilter(AuthenticationManager authenticationManager) { + this.authenticationManager = authenticationManager; + } + + + + @Override + public Authentication attemptAuthentication(HttpServletRequest request, HttpServletResponse response) + throws AuthenticationException { + try { + log.info("로그인 요청 들어"); + + // 요청 본문을 UserDto로 바로 변환 + UserDto userDto = new ObjectMapper().readValue(request.getInputStream(), UserDto.class); + + if (userDto.getUsername() == null || userDto.getPassword() == null) { + throw new IllegalArgumentException("uid와 password가 필요합니다."); + } + + // 인증 토큰 생성 + UsernamePasswordAuthenticationToken authenticationToken = + new UsernamePasswordAuthenticationToken(userDto.getUsername(), userDto.getPassword()); + + // 인증을 처리하고 인증 성공 후에 반환된 Authentication 객체 + Authentication authentication = authenticationManager.authenticate(authenticationToken); + + return authentication; + } catch (IOException e) { + log.error("요청 본문 파싱 오류", e); + throw new IllegalArgumentException("요청 데이터를 처리할 수 없습니다."); + } + } + + @Override + protected void successfulAuthentication(HttpServletRequest request, + HttpServletResponse response, + FilterChain chain, + Authentication authResult) throws IOException, ServletException { + log.info("successfulAuthentication() 진입"); + + PrincipalDetails principalDetails = (PrincipalDetails) authResult.getPrincipal(); + log.info("PrincipalDetails: {}", principalDetails); + + // JWT 토큰 생성 + String jwtToken = JWT.create() + .withSubject(principalDetails.getUsername()) + .withExpiresAt(new Date(System.currentTimeMillis() + JwtProperties.EXPIRATION_TIME)) + .withClaim("uid", principalDetails.getUser().getUsername()) + .sign(Algorithm.HMAC512(JwtProperties.SECRET)); + + // 토큰을 응답 본문으로 JSON 형태로 반환 + response.setContentType("application/json"); + response.getWriter().write("{\"token\":\"" + jwtToken + "\", \"redirect\":\"/\"}"); // 리다이렉트 정보와 토큰 함께 반환 + } + + + +} diff --git a/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java new file mode 100644 index 0000000..e9898eb --- /dev/null +++ b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java @@ -0,0 +1,78 @@ +package com.recipe.cookofking.config.jwt; + + +import java.io.IOException; + +import org.springframework.security.authentication.AuthenticationManager; +import org.springframework.security.authentication.UsernamePasswordAuthenticationToken; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.web.authentication.www.BasicAuthenticationFilter; + +import com.recipe.cookofking.config.auth.PrincipalDetails; +import com.recipe.cookofking.entity.User; +import com.recipe.cookofking.repository.UserRepository; + +import jakarta.servlet.FilterChain; +import jakarta.servlet.ServletException; +import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpServletResponse; +import lombok.extern.slf4j.Slf4j; +import software.amazon.awssdk.core.checksums.Algorithm; + +@Slf4j +public class JwtAuthorizationFilter extends BasicAuthenticationFilter { + + private final UserRepository userRepository; + + + public JwtAuthorizationFilter(AuthenticationManager authenticationManager, UserRepository userRepository) { + super(authenticationManager); + this.userRepository = userRepository; + } + + + // 로그인요청ㅊ + @Override + protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain chain) + throws IOException, ServletException { + + // 헤더에서 Authorization 키 확인 + String header = request.getHeader("Authorization"); + + if (header == null || !header.startsWith("Bearer ")) { + chain.doFilter(request, response); + return; + } + // Authorization 헤더에서 토큰 추출 + String token = header.replace("Bearer ", ""); + + try { + String username = JWT.require(Algorithm.HMAC512(JwtProperties.SECRET)) + .build() + .verify(token) + .getClaim("username") + .asString(); + + if (username != null) { + User user = userRepository.findBy(username) + .orElseThrow(() -> new RuntimeException("User not found")); + PrincipalDetails principalDetails = new PrincipalDetails(user); + + Authentication authentication = new UsernamePasswordAuthenticationToken( + principalDetails, null, principalDetails.getAuthorities()); + + SecurityContextHolder.getContext().setAuthentication(authentication); + } + } catch (Exception e) { + log.error("JWT 처리 중 오류 발생", e); + response.setStatus(HttpServletResponse.SC_FORBIDDEN); + response.getWriter().write("Forbidden: Invalid or expired token"); + return; + } + + chain.doFilter(request, response); + } + + +} diff --git a/src/main/java/com/recipe/cookofking/config/jwt/JwtProperties.java b/src/main/java/com/recipe/cookofking/config/jwt/JwtProperties.java new file mode 100644 index 0000000..3a50d12 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/config/jwt/JwtProperties.java @@ -0,0 +1,9 @@ +package com.recipe.cookofking.config.jwt; + +public interface JwtProperties { + + String SECRET = "jwtSecret"; + int EXPIRATION_TIME = 60*60*24*1000; + String TOKEN_PREFIX = "Bearer "; + String HEADER_STRING = "Authorization"; +} diff --git a/src/main/java/com/recipe/cookofking/controller/PostController.java b/src/main/java/com/recipe/cookofking/controller/PostController.java index e239f71..c9d92a9 100644 --- a/src/main/java/com/recipe/cookofking/controller/PostController.java +++ b/src/main/java/com/recipe/cookofking/controller/PostController.java @@ -3,7 +3,10 @@ import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; +import lombok.RequiredArgsConstructor; + @Controller +@RequiredArgsConstructor public class PostController { @RequestMapping("/") diff --git a/src/main/java/com/recipe/cookofking/controller/UserController.java b/src/main/java/com/recipe/cookofking/controller/UserController.java index 68c3a38..97fee4e 100644 --- a/src/main/java/com/recipe/cookofking/controller/UserController.java +++ b/src/main/java/com/recipe/cookofking/controller/UserController.java @@ -1,12 +1,119 @@ package com.recipe.cookofking.controller; +import java.util.HashMap; +import java.util.Map; + +import org.springframework.http.HttpHeaders; +import org.springframework.http.HttpStatus; +import org.springframework.http.ResponseEntity; +import org.springframework.stereotype.Controller; +import org.springframework.ui.Model; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.ModelAttribute; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.bind.annotation.RequestParam; +import org.springframework.web.bind.annotation.ResponseBody; + +import com.recipe.cookofking.dto.UserDto; +import com.recipe.cookofking.dto.request.UserRequest; +import com.recipe.cookofking.service.UserService; -@RestController -@RequestMapping("/users") +import lombok.RequiredArgsConstructor; +import lombok.extern.slf4j.Slf4j; + +@Controller +@RequiredArgsConstructor +@RequestMapping("/user") +@Slf4j public class UserController { + + private final UserService userService; + + + + /* 로그인 페이지 */ + @GetMapping("/login") + public String login( ) { + log.info("로그인 페이지 !"); + + return "user/login"; // 로그인 페이지로 이동 + } + + /* 로그인 기능 */ + @PostMapping("/login") + @ResponseBody + public ResponseEntity> login(@RequestBody UserRequest userRequest) { + log.info("로그인 시도: {}", userRequest.getUsername()); + + try { + // UserRequest에서 UserDto로 변환 + UserDto userDto = userRequest.toDto(); + + + // 사용자 인증 및 JWT 토큰 발급 + String token = userService.authenticateUser(userDto); // JWT 토큰 발급 + + // 로그인 성공 시 메시지 및 토큰 반환 + Map response = new HashMap<>(); + response.put("message", "로그인 성공!"); + response.put("token", token); // 토큰을 응답 본문에 포함시킴 + + // 200 OK 응답, 헤더에 Authorization 포함 + HttpHeaders headers = new HttpHeaders(); + headers.set("Authorization", "Bearer " + token); // 헤더에 토큰 추가 + return ResponseEntity.ok() + .headers(headers) + .body(response); // 응답 본문과 헤더 모두 반환 + } catch (Exception e) { + // 로그인 실패 시 메시지 반환 + Map response = new HashMap<>(); + response.put("message", "로그인 실패: " + e.getMessage()); + + return ResponseEntity.status(HttpStatus.UNAUTHORIZED) // 401 Unauthorized + .body(response); + } + } + + + /* 회원가입 페이지 이동*/ + @GetMapping("/register") + public String userregister() { + return "/user/register"; + } + + + /* 회원가입 기능 */ + @PostMapping("/register") + public String signup(@ModelAttribute UserRequest userRequest, Model model) { + try { + + // UserRequest를 UserDto로 변환 + UserDto userDto= userRequest.toDto(); + + userService.registerUser(userDto); + log.info("회원가입에서 입력한 값 : " + userRequest); // 로그 추가 + log.info("디비에저장되는값 : " + userDto); // 로그 추가 + + return "redirect:/user/login"; + + }catch(IllegalArgumentException e) { + model.addAttribute("errorMessage", e.getMessage()); // 중복 아이디 메시지 + return "/user/register"; // 회원가입 페이지로 돌아감 + }catch(Exception e) { + System.out.println("회원가입 오류: " + e.getMessage()); + model.addAttribute("errorMessage", "회원가입에 실패했습니다: " + e.getMessage()); + return "/user/register"; // 오류 발생 시 회원가입 페이지로 복귀 + } + } + + @GetMapping("/mypage") + public String myPage() { + return "user/mypage"; + } + } diff --git a/src/main/java/com/recipe/cookofking/dto/UserDto.java b/src/main/java/com/recipe/cookofking/dto/UserDto.java index 14e9bc3..48087b5 100644 --- a/src/main/java/com/recipe/cookofking/dto/UserDto.java +++ b/src/main/java/com/recipe/cookofking/dto/UserDto.java @@ -1,6 +1,8 @@ package com.recipe.cookofking.dto; import com.recipe.cookofking.entity.User; + + import lombok.*; import java.io.Serializable; @@ -14,12 +16,53 @@ @NoArgsConstructor @Builder public class UserDto implements Serializable { - Integer id; - String username; + + private static final long serialVersionUID = 1L; + Integer id; + String username; String email; String password; String role; LocalDateTime createdDate; LocalDateTime modifiedDate; + + public UserDto(String username, String password) { + this.username = username; + this.password = password; + } + + public UserDto(String username, String email, String password, String role) { + this.username = username; + this.email = email; + this.password = password; + this.role = role; + this.createdDate = null; // 기본값 null + this.modifiedDate = null; // 기본값 null + } + + + // 생성자 추가 + public static UserDto of( String username, String password, String email, String role,LocalDateTime createdDate,LocalDateTime modifiedDate) { + return UserDto.of(username, password, email, role, null, null); + } + + + // Entity -> DTO 변환 메소드 + public static UserDto fromEntity(User user) { + return UserDto.builder() + .id(user.getId()) + .username(user.getUsername()) + .email(user.getEmail()) + .password(user.getPassword()) + .role(user.getRole()) + .createdDate(user.getCreatedDate()) + .modifiedDate(user.getModifiedDate()) + .build(); + } + + // UserDto를 엔티티로 변환하는 메소드 + public User toEntity() { + return new User( username, email, password, role, null, null); + } } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/dto/request/UserRequest.java b/src/main/java/com/recipe/cookofking/dto/request/UserRequest.java new file mode 100644 index 0000000..0b0ef96 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/dto/request/UserRequest.java @@ -0,0 +1,37 @@ +package com.recipe.cookofking.dto.request; + +import java.time.LocalDateTime; + +import com.recipe.cookofking.dto.UserDto; + +import lombok.AllArgsConstructor; +import lombok.Getter; +import lombok.NoArgsConstructor; +import lombok.Setter; +import lombok.ToString; + +@AllArgsConstructor +@NoArgsConstructor +@Getter +@Setter +@ToString +public class UserRequest { + + private String username; + private String email; + private String password; + private String role; + private LocalDateTime createdDate; + private LocalDateTime modifiedDate; + + + // UserDto로 변환하는 메소드 + public UserDto toDto() { + if (role == null) { + role = "ROLE_USER"; // 기본값 설정 + } + // 생성자에서 createdDate와 modifiedDate를 null로 설정 + return new UserDto(username, email, password, role); + } +} + diff --git a/src/main/java/com/recipe/cookofking/entity/User.java b/src/main/java/com/recipe/cookofking/entity/User.java index 57bf878..f4e6817 100644 --- a/src/main/java/com/recipe/cookofking/entity/User.java +++ b/src/main/java/com/recipe/cookofking/entity/User.java @@ -43,5 +43,15 @@ public class User { @Column(name = "modifiedDate") @LastModifiedDate private LocalDateTime modifiedDate; + + // User 엔티티 생성 + public User(String username, String email, String password, String role, LocalDateTime createdDate, LocalDateTime modifiedDate) { + this.username = username; + this.email = email; + this.password = password; + this.role = role; + this.createdDate = createdDate; + this.modifiedDate = modifiedDate; + } } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/repository/UserRepository.java b/src/main/java/com/recipe/cookofking/repository/UserRepository.java index 69e8f62..4a83574 100644 --- a/src/main/java/com/recipe/cookofking/repository/UserRepository.java +++ b/src/main/java/com/recipe/cookofking/repository/UserRepository.java @@ -1,10 +1,16 @@ package com.recipe.cookofking.repository; -import com.recipe.cookofking.entity.User; +import java.util.Optional; + import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.stereotype.Repository; +import com.recipe.cookofking.entity.User; + @Repository public interface UserRepository extends JpaRepository { - // Add custom query methods if needed + + Optional findByUsername(String username); + boolean existsByUsername(String username); // username 중복 확인 + } diff --git a/src/main/java/com/recipe/cookofking/service/UserService.java b/src/main/java/com/recipe/cookofking/service/UserService.java index ffda778..2328b6d 100644 --- a/src/main/java/com/recipe/cookofking/service/UserService.java +++ b/src/main/java/com/recipe/cookofking/service/UserService.java @@ -1,6 +1,74 @@ package com.recipe.cookofking.service; +import java.util.Date; + +import org.springframework.security.authentication.AuthenticationManager; +import org.springframework.security.authentication.UsernamePasswordAuthenticationToken; +import org.springframework.security.core.Authentication; +import org.springframework.security.crypto.password.PasswordEncoder; import org.springframework.stereotype.Service; + + +import com.auth0.jwt.JWT; +import com.auth0.jwt.algorithms.Algorithm; + +import com.recipe.cookofking.config.jwt.JwtProperties; +import com.recipe.cookofking.dto.UserDto; +import com.recipe.cookofking.entity.User; +import com.recipe.cookofking.repository.UserRepository; + + +import lombok.RequiredArgsConstructor; +import lombok.extern.slf4j.Slf4j; + + @Service +@RequiredArgsConstructor +@Slf4j public class UserService { - // Add User related business logic here + + + private final UserRepository userRepository; + private final PasswordEncoder passwordEncoder; + private final AuthenticationManager authenticationManager; + + // 사용자 인증 후 JWT 토큰 발급 + public String authenticateUser(UserDto userDto) throws Exception { + // 인증 토큰 생성 + UsernamePasswordAuthenticationToken authenticationToken = + new UsernamePasswordAuthenticationToken(userDto.getUsername(), userDto.getPassword()); + + // 인증 처리 + Authentication authentication = authenticationManager.authenticate(authenticationToken); + + // 인증 성공 후 JWT 토큰 생성 + String jwtToken = JWT.create() + .withSubject(authentication.getName()) // 사용자 이름 + .withExpiresAt(new Date(System.currentTimeMillis() + JwtProperties.EXPIRATION_TIME)) // 만료 시간 + .withClaim("uid", userDto.getUsername()) // 추가적인 claim, 예: 사용자 ID + .sign(Algorithm.HMAC512(JwtProperties.SECRET)); // 비밀 키로 서명 + + return jwtToken; + } + + + public void registerUser(UserDto userDto) { + + // 아이디 중복체크 + if(userRepository.existsByUsername(userDto.getUsername())) { + throw new IllegalArgumentException("이미 존재하는 아이디입니다."); + } + // 암호화 추가 + String rawPassword = userDto.getPassword(); + String encryptedPassword = passwordEncoder.encode(rawPassword); + userDto.setPassword(encryptedPassword); + + //userDto를 user 엔티티로 변환 + User user = userDto.toEntity(); + userRepository.save(user); + + } + + + + } \ No newline at end of file diff --git a/src/main/resources/static/css/register.css b/src/main/resources/static/css/register.css new file mode 100644 index 0000000..a981d9b --- /dev/null +++ b/src/main/resources/static/css/register.css @@ -0,0 +1,103 @@ +@charset "UTF-8"; +html, +body { + height: 100%; +} + +.form-signin { + max-width: 330px; + padding: 1rem; +} + +.form-signin .form-floating:focus-within { + z-index: 2; +} + +.form-signin input[type="email"] { + margin-bottom: -1px; + border-bottom-right-radius: 0; + border-bottom-left-radius: 0; +} + +.form-signin input[type="password"] { + margin-bottom: 10px; + border-top-left-radius: 0; + border-top-right-radius: 0; +} + + .bd-placeholder-img { + font-size: 1.125rem; + text-anchor: middle; + -webkit-user-select: none; + -moz-user-select: none; + user-select: none; + } + + @media (min-width: 768px) { + .bd-placeholder-img-lg { + font-size: 3.5rem; + } + } + + .b-example-divider { + width: 100%; + height: 3rem; + background-color: rgba(0, 0, 0, .1); + border: solid rgba(0, 0, 0, .15); + border-width: 1px 0; + box-shadow: inset 0 .5em 1.5em rgba(0, 0, 0, .1), inset 0 .125em .5em rgba(0, 0, 0, .15); + } + + .b-example-vr { + flex-shrink: 0; + width: 1.5rem; + height: 100vh; + } + + .bi { + vertical-align: -.125em; + fill: currentColor; + } + + .nav-scroller { + position: relative; + z-index: 2; + height: 2.75rem; + overflow-y: hidden; + } + + .nav-scroller .nav { + display: flex; + flex-wrap: nowrap; + padding-bottom: 1rem; + margin-top: -1px; + overflow-x: auto; + text-align: center; + white-space: nowrap; + -webkit-overflow-scrolling: touch; + } + + .btn-bd-primary { + --bd-violet-bg: #712cf9; + --bd-violet-rgb: 112.520718, 44.062154, 249.437846; + + --bs-btn-font-weight: 600; + --bs-btn-color: var(--bs-white); + --bs-btn-bg: var(--bd-violet-bg); + --bs-btn-border-color: var(--bd-violet-bg); + --bs-btn-hover-color: var(--bs-white); + --bs-btn-hover-bg: #6528e0; + --bs-btn-hover-border-color: #6528e0; + --bs-btn-focus-shadow-rgb: var(--bd-violet-rgb); + --bs-btn-active-color: var(--bs-btn-hover-color); + --bs-btn-active-bg: #5a23c8; + --bs-btn-active-border-color: #5a23c8; + } + + .bd-mode-toggle { + z-index: 1500; + } + + .bd-mode-toggle .dropdown-menu .active .bi { + display: block !important; + } diff --git a/src/main/resources/templates/user/login.html b/src/main/resources/templates/user/login.html new file mode 100644 index 0000000..b7ee086 --- /dev/null +++ b/src/main/resources/templates/user/login.html @@ -0,0 +1,22 @@ + + + + +Insert title here + + + + + + + + +
    + + + +
    + + + \ No newline at end of file diff --git a/src/main/resources/templates/user/register.html b/src/main/resources/templates/user/register.html new file mode 100644 index 0000000..8348d08 --- /dev/null +++ b/src/main/resources/templates/user/register.html @@ -0,0 +1,62 @@ + + + + + + + Signin Template · Bootstrap v5.3 + + + + + + + + + + + + + + + + + + + + From 032f62f219cdeda5c6fcad19084a441cd8e32b32 Mon Sep 17 00:00:00 2001 From: SUJINJEONG012 Date: Mon, 10 Feb 2025 14:28:59 +0900 Subject: [PATCH 09/73] =?UTF-8?q?jwt=20=EB=B6=80=EB=B6=84=EC=A0=81=20?= =?UTF-8?q?=EC=BD=94=EB=93=9C=20=EC=83=9D=EC=84=B1?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../cookofking/config/SecurityConfig.java | 75 ++++++++++++++++--- .../config/jwt/JwtAuthenticationFilter.java | 3 +- .../config/jwt/JwtAuthorizationFilter.java | 6 +- 3 files changed, 69 insertions(+), 15 deletions(-) diff --git a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java index b101194..b60b244 100644 --- a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java +++ b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java @@ -7,25 +7,78 @@ import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.security.authentication.AuthenticationManager; +import org.springframework.security.config.annotation.authentication.builders.AuthenticationManagerBuilder; import org.springframework.security.config.annotation.web.builders.HttpSecurity; +import org.springframework.security.config.annotation.web.configuration.EnableWebSecurity; +import org.springframework.security.config.annotation.web.configurers.AbstractHttpConfigurer; +import org.springframework.security.config.http.SessionCreationPolicy; import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; import org.springframework.security.crypto.password.PasswordEncoder; import org.springframework.security.web.SecurityFilterChain; +import com.recipe.cookofking.config.jwt.JwtAuthenticationFilter; +import com.recipe.cookofking.config.jwt.JwtAuthorizationFilter; +import com.recipe.cookofking.repository.UserRepository; + +import lombok.RequiredArgsConstructor; + +@RequiredArgsConstructor @Configuration +@EnableWebSecurity public class SecurityConfig { - @Bean - public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception { - http - .authorizeHttpRequests(auth -> auth - .anyRequest().permitAll() // 모든 요청 허용 - ) - .csrf(csrf -> csrf.disable()) // CSRF 비활성화 (개발 환경) - .formLogin(form -> form.disable()); // 폼 로그인 비활성화 - - return http.build(); - } + private final UserRepository userRepository; + + @Bean + public AuthenticationManager authenticationManager(HttpSecurity http) throws Exception { + return http.getSharedObject(AuthenticationManagerBuilder.class).build(); + } + +// @Bean +// public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception { +// http +// .authorizeHttpRequests(auth -> auth +// .anyRequest().permitAll() // 모든 요청 허용 +// ) +// .csrf(csrf -> csrf.disable()) // CSRF 비활성화 (개발 환경) +// .formLogin(form -> form.disable()); // 폼 로그인 비활성화 +// +// return http.build(); +// } + + @Bean + public SecurityFilterChain filterChain(HttpSecurity http, AuthenticationManager authenticationManager ) throws Exception { + + http + // 로그인 시 JWT 토큰 발급을 위한 필터 추가 (JwtAuthenticationFilter) + .addFilter(new JwtAuthenticationFilter(authenticationManager)) + + // 요청 시 JWT 토큰을 검증하는 필터 추가 (JwtAuthorizationFilter) + .addFilter(new JwtAuthorizationFilter(authenticationManager, userRepository)) + + .csrf(AbstractHttpConfigurer::disable) + + .sessionManagement(session -> session + .sessionCreationPolicy(SessionCreationPolicy.STATELESS)) + + .formLogin(AbstractHttpConfigurer::disable) + .httpBasic(AbstractHttpConfigurer::disable) + + // 경로별 권한 설정 + .authorizeHttpRequests(auth -> auth + .requestMatchers("/api/**","/**", "/login","/register","/js/**", "/css/**", "/images/**", "/static/**").permitAll() // 로그인 경로는 인증 없이 접근 가능 + .requestMatchers("/api/mypage").authenticated() // 🔹 마이페이지는 인증 필요 + + .requestMatchers("/admin/**").hasRole("ADMIN") // 예시: admin 권한이 필요한 경로 + .anyRequest().authenticated()); // 다른 모든 요청은 인증 필요 + + return http.build(); + } + + + + @Bean public PasswordEncoder passwordEncoder() { diff --git a/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthenticationFilter.java b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthenticationFilter.java index 70867dc..3377ae1 100644 --- a/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthenticationFilter.java +++ b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthenticationFilter.java @@ -11,6 +11,7 @@ import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter; import com.auth0.jwt.JWT; +import com.auth0.jwt.algorithms.Algorithm; import com.fasterxml.jackson.databind.ObjectMapper; import com.recipe.cookofking.config.auth.PrincipalDetails; import com.recipe.cookofking.dto.UserDto; @@ -20,8 +21,6 @@ import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; import lombok.extern.slf4j.Slf4j; -import software.amazon.awssdk.core.checksums.Algorithm; - // 로그인처리 담당하는 필터 @Slf4j diff --git a/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java index e9898eb..b21e0e6 100644 --- a/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java +++ b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java @@ -9,6 +9,9 @@ import org.springframework.security.core.context.SecurityContextHolder; import org.springframework.security.web.authentication.www.BasicAuthenticationFilter; +import com.auth0.jwt.JWT; +import com.auth0.jwt.algorithms.Algorithm; + import com.recipe.cookofking.config.auth.PrincipalDetails; import com.recipe.cookofking.entity.User; import com.recipe.cookofking.repository.UserRepository; @@ -18,7 +21,6 @@ import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; import lombok.extern.slf4j.Slf4j; -import software.amazon.awssdk.core.checksums.Algorithm; @Slf4j public class JwtAuthorizationFilter extends BasicAuthenticationFilter { @@ -55,7 +57,7 @@ protected void doFilterInternal(HttpServletRequest request, HttpServletResponse .asString(); if (username != null) { - User user = userRepository.findBy(username) + User user = userRepository.findByUsername(username) .orElseThrow(() -> new RuntimeException("User not found")); PrincipalDetails principalDetails = new PrincipalDetails(user); From 45d03f6f47b92b1039e248ed0d6f2a797af7d38a Mon Sep 17 00:00:00 2001 From: SUJINJEONG012 Date: Mon, 10 Feb 2025 15:08:34 +0900 Subject: [PATCH 10/73] feature: register,login page & fix register, login, jwt, security --- .../cookofking/config/SecurityConfig.java | 9 +- .../cookofking/controller/MainController.java | 11 +++ src/main/resources/templates/index.html | 12 +++ src/main/resources/templates/user/login.html | 88 ++++++++++++++++--- .../resources/templates/user/register.html | 1 + 5 files changed, 103 insertions(+), 18 deletions(-) create mode 100644 src/main/java/com/recipe/cookofking/controller/MainController.java create mode 100644 src/main/resources/templates/index.html diff --git a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java index b60b244..b93e90a 100644 --- a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java +++ b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java @@ -1,10 +1,6 @@ package com.recipe.cookofking.config; -//@EnableMethodSecurity(prePostEnabled = true,securedEnabled = true) -//@EnableWebSecurity -//@Configuration - import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.security.authentication.AuthenticationManager; @@ -68,9 +64,8 @@ public SecurityFilterChain filterChain(HttpSecurity http, AuthenticationManager // 경로별 권한 설정 .authorizeHttpRequests(auth -> auth .requestMatchers("/api/**","/**", "/login","/register","/js/**", "/css/**", "/images/**", "/static/**").permitAll() // 로그인 경로는 인증 없이 접근 가능 - .requestMatchers("/api/mypage").authenticated() // 🔹 마이페이지는 인증 필요 - - .requestMatchers("/admin/**").hasRole("ADMIN") // 예시: admin 권한이 필요한 경로 + //.requestMatchers("/api/mypage").authenticated() // 인증필요시 + //.requestMatchers("/admin/**").hasRole("ADMIN") // 예시: admin 권한이 필요한 경로 .anyRequest().authenticated()); // 다른 모든 요청은 인증 필요 return http.build(); diff --git a/src/main/java/com/recipe/cookofking/controller/MainController.java b/src/main/java/com/recipe/cookofking/controller/MainController.java new file mode 100644 index 0000000..5db6b55 --- /dev/null +++ b/src/main/java/com/recipe/cookofking/controller/MainController.java @@ -0,0 +1,11 @@ +package com.recipe.cookofking.controller; + +import org.springframework.stereotype.Controller; + +@Controller +public class MainController { + + public String main() { + return "/index"; + } +} diff --git a/src/main/resources/templates/index.html b/src/main/resources/templates/index.html new file mode 100644 index 0000000..a5fce8a --- /dev/null +++ b/src/main/resources/templates/index.html @@ -0,0 +1,12 @@ + + + + + +Insert title here + + +메인페이지 + + \ No newline at end of file diff --git a/src/main/resources/templates/user/login.html b/src/main/resources/templates/user/login.html index b7ee086..cf34192 100644 --- a/src/main/resources/templates/user/login.html +++ b/src/main/resources/templates/user/login.html @@ -1,22 +1,88 @@ - - -Insert title here - - + + + 로그인 + + + -
    - - - -
    + + + + + + + + + + + + + - \ No newline at end of file + diff --git a/src/main/resources/templates/user/register.html b/src/main/resources/templates/user/register.html index 8348d08..87c0c8d 100644 --- a/src/main/resources/templates/user/register.html +++ b/src/main/resources/templates/user/register.html @@ -8,6 +8,7 @@ Signin Template · Bootstrap v5.3 + From 53176c21de10802205d23bab2e84856f7fe63c07 Mon Sep 17 00:00:00 2001 From: KangSeongKwan Date: Mon, 10 Feb 2025 13:33:03 +0900 Subject: [PATCH 11/73] Make View Like but Returned 500 Error Version --- .../cookofking/controller/LikeController.java | 22 ++++++++++++++++++- .../com/recipe/cookofking/entity/Like.java | 2 -- .../cookofking/repository/LikeRepository.java | 1 + .../cookofking/repository/PostRepository.java | 2 ++ .../cookofking/service/LikeService.java | 21 ++++++++++++++++++ 5 files changed, 45 insertions(+), 3 deletions(-) diff --git a/src/main/java/com/recipe/cookofking/controller/LikeController.java b/src/main/java/com/recipe/cookofking/controller/LikeController.java index e1844e9..08dc0e3 100644 --- a/src/main/java/com/recipe/cookofking/controller/LikeController.java +++ b/src/main/java/com/recipe/cookofking/controller/LikeController.java @@ -1,10 +1,30 @@ package com.recipe.cookofking.controller; +import com.recipe.cookofking.dto.LikeDto; +import com.recipe.cookofking.service.LikeService; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestParam; import org.springframework.web.bind.annotation.RestController; + @RestController -@RequestMapping("/likes") +@RequestMapping("/api/v1/likes") public class LikeController { + private final LikeService likeService; + + public LikeController(LikeService likeService) { + this.likeService = likeService; + } + @GetMapping + public ResponseEntity getLikes(@RequestParam Integer postId) { + try { + LikeDto responseDto = likeService.getLikes(postId); + return ResponseEntity.ok(responseDto); + } catch (IllegalArgumentException e) { + return ResponseEntity.badRequest().body(e.getMessage()); + } + } } diff --git a/src/main/java/com/recipe/cookofking/entity/Like.java b/src/main/java/com/recipe/cookofking/entity/Like.java index 0c23e89..bf77710 100644 --- a/src/main/java/com/recipe/cookofking/entity/Like.java +++ b/src/main/java/com/recipe/cookofking/entity/Like.java @@ -31,6 +31,4 @@ public class Like { @Column(name = "createdDate") @CreatedDate private LocalDateTime createdDate; - - } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/repository/LikeRepository.java b/src/main/java/com/recipe/cookofking/repository/LikeRepository.java index 32a3b9f..749c434 100644 --- a/src/main/java/com/recipe/cookofking/repository/LikeRepository.java +++ b/src/main/java/com/recipe/cookofking/repository/LikeRepository.java @@ -6,4 +6,5 @@ @Repository public interface LikeRepository extends JpaRepository { // Add custom query methods if needed + long countByPostId(Integer postId); // 특정 게시물의 좋아요 개수를 반환 } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/repository/PostRepository.java b/src/main/java/com/recipe/cookofking/repository/PostRepository.java index 6469422..4a24eb8 100644 --- a/src/main/java/com/recipe/cookofking/repository/PostRepository.java +++ b/src/main/java/com/recipe/cookofking/repository/PostRepository.java @@ -1,7 +1,9 @@ package com.recipe.cookofking.repository; + import com.recipe.cookofking.entity.Post; import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.stereotype.Repository; + @Repository public interface PostRepository extends JpaRepository { // Add custom query methods if needed diff --git a/src/main/java/com/recipe/cookofking/service/LikeService.java b/src/main/java/com/recipe/cookofking/service/LikeService.java index adb277e..56d8cdc 100644 --- a/src/main/java/com/recipe/cookofking/service/LikeService.java +++ b/src/main/java/com/recipe/cookofking/service/LikeService.java @@ -1,6 +1,27 @@ package com.recipe.cookofking.service; + +import com.recipe.cookofking.dto.LikeDto; +import com.recipe.cookofking.repository.LikeRepository; +import com.recipe.cookofking.repository.PostRepository; import org.springframework.stereotype.Service; + @Service public class LikeService { // Add Like related business logic here + private final LikeRepository likeRepository; + private final PostRepository postRepository; + + public LikeService(LikeRepository likeRepository, PostRepository postRepository) { + this.likeRepository = likeRepository; + this.postRepository = postRepository; + } + + public LikeDto getLikes(Integer postId) { + if (!postRepository.existsById(postId)) { + throw new IllegalArgumentException("Post not found"); + } + + long likesCount = likeRepository.countByPostId(postId); + return new LikeDto(); + } } From a832df2516922493c7f00be4387a6d46603761a0 Mon Sep 17 00:00:00 2001 From: phc979 Date: Mon, 10 Feb 2025 15:58:47 +0900 Subject: [PATCH 12/73] feature : post-list --- .../cookofking/controller/PostController.java | 39 +++++++++++++++++++ .../recipe/cookofking/dto/post/PostDto.java | 4 ++ .../com/recipe/cookofking/entity/Post.java | 25 ++++++++++++ .../recipe/cookofking/mapper/PostMapper.java | 4 ++ 4 files changed, 72 insertions(+) diff --git a/src/main/java/com/recipe/cookofking/controller/PostController.java b/src/main/java/com/recipe/cookofking/controller/PostController.java index 0cc654b..8f76406 100644 --- a/src/main/java/com/recipe/cookofking/controller/PostController.java +++ b/src/main/java/com/recipe/cookofking/controller/PostController.java @@ -3,10 +3,15 @@ import com.recipe.cookofking.dto.post.PostDto; import com.recipe.cookofking.service.PostService; import lombok.RequiredArgsConstructor; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.PageRequest; +import org.springframework.data.domain.Sort; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; +import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PathVariable; import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestParam; @Controller @RequiredArgsConstructor @@ -30,4 +35,38 @@ public String showViewForm(@PathVariable Integer postid, Model model) { return "Post/post-view"; // post-view.html 렌더링 } + + @GetMapping("/list") + public String getPostList(Model model, + @RequestParam(defaultValue = "0") int page, + @RequestParam(defaultValue = "20") int size, + @RequestParam(defaultValue = "latest") String sort) { + + // 정렬 기준 설정 + Sort sortOrder = getSortOrder(sort); + PageRequest pageRequest = PageRequest.of(page, size, sortOrder); + + // 서비스 호출 + Page postPage = postService.getPostList(pageRequest); + + // 모델에 데이터 추가 + model.addAttribute("postPage", postPage); + model.addAttribute("totalPosts", postPage.getTotalElements()); + model.addAttribute("currentSort", sort); // 현재 정렬 기준을 모델에 추가 (버튼 하이라이트용) + + return "Post/posts"; // posts.html로 렌더링 + } + + // 정렬 기준에 따라 Sort 객체 반환 + private Sort getSortOrder(String sort) { + switch (sort) { + case "views": + return Sort.by(Sort.Direction.DESC, "viewCount"); // 조회수 기준 정렬 + case "likes": + return Sort.by(Sort.Direction.DESC, "likeCount"); // 좋아요 기준 정렬 + default: + return Sort.by(Sort.Direction.DESC, "createdDate"); // 기본값: 최신순 정렬 + } + } + } diff --git a/src/main/java/com/recipe/cookofking/dto/post/PostDto.java b/src/main/java/com/recipe/cookofking/dto/post/PostDto.java index bdab910..dfb9030 100644 --- a/src/main/java/com/recipe/cookofking/dto/post/PostDto.java +++ b/src/main/java/com/recipe/cookofking/dto/post/PostDto.java @@ -2,6 +2,7 @@ import com.recipe.cookofking.dto.UserDto; import com.recipe.cookofking.entity.Post; +import jakarta.persistence.Column; import lombok.*; import java.io.Serializable; @@ -22,6 +23,9 @@ public class PostDto implements Serializable { String ingredients; String instructions; String mainImageS3URL; + Integer viewCount; + Integer likeCount; LocalDateTime createdDate; LocalDateTime modifiedDate; + } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/entity/Post.java b/src/main/java/com/recipe/cookofking/entity/Post.java index 2a4daaa..ea4cb55 100644 --- a/src/main/java/com/recipe/cookofking/entity/Post.java +++ b/src/main/java/com/recipe/cookofking/entity/Post.java @@ -45,6 +45,12 @@ public class Post { @Column(name = "mainImageS3URL") private String mainImageS3URL; + @Column(name = "view_count", nullable = false) + private Integer viewCount = 0; + + @Column(name = "like_count", nullable = false) + private Integer likeCount = 0; + @Column(name = "createdDate") @CreatedDate private LocalDateTime createdDate; @@ -53,4 +59,23 @@ public class Post { @LastModifiedDate private LocalDateTime modifiedDate; + // 조회수 감소 메서드 + public void decrementViewCount() { + this.viewCount -= 1; + } + + // 조회수 증가 메서드 + public void incrementViewCount() { + this.viewCount += 1; + } + + // 좋아요 수 감소 메서드 (필요 시) + public void decrementLikeCount() { + this.likeCount -= 1; + } + + // 좋아요 수 증가 메서드 (필요 시) + public void incrementLikeCount() { + this.likeCount += 1; + } } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/mapper/PostMapper.java b/src/main/java/com/recipe/cookofking/mapper/PostMapper.java index 4222da6..ef3cfc2 100644 --- a/src/main/java/com/recipe/cookofking/mapper/PostMapper.java +++ b/src/main/java/com/recipe/cookofking/mapper/PostMapper.java @@ -16,6 +16,8 @@ public static PostDto toDto(Post post) { .ingredients(post.getIngredients()) .instructions(post.getInstructions()) .mainImageS3URL(post.getMainImageS3URL()) + .viewCount(post.getViewCount()) // 추가 + .likeCount(post.getLikeCount()) // 추가 .createdDate(post.getCreatedDate()) .modifiedDate(post.getModifiedDate()) .build(); @@ -33,6 +35,8 @@ public static Post toEntity(PostDto postDto) { .ingredients(postDto.getIngredients()) .instructions(postDto.getInstructions()) .mainImageS3URL(postDto.getMainImageS3URL()) + .viewCount(postDto.getViewCount()) // 추가 + .likeCount(postDto.getLikeCount()) // 추가 .createdDate(postDto.getCreatedDate()) .modifiedDate(postDto.getModifiedDate()) .build(); From dceba358ce2187a17342af89cbd21c8aae733996 Mon Sep 17 00:00:00 2001 From: SUJINJEONG012 Date: Mon, 10 Feb 2025 16:16:03 +0900 Subject: [PATCH 13/73] feature: layouts with Thymeleaf --- .../resources/templates/fragments/body.html | 107 ++++++++++++++++++ .../resources/templates/fragments/footer.html | 29 +++++ .../resources/templates/fragments/header.html | 18 +++ .../resources/templates/layout/basic.html | 6 + src/main/resources/templates/user/login.html | 49 ++++---- 5 files changed, 189 insertions(+), 20 deletions(-) create mode 100644 src/main/resources/templates/fragments/body.html create mode 100644 src/main/resources/templates/fragments/footer.html create mode 100644 src/main/resources/templates/fragments/header.html create mode 100644 src/main/resources/templates/layout/basic.html diff --git a/src/main/resources/templates/fragments/body.html b/src/main/resources/templates/fragments/body.html new file mode 100644 index 0000000..bea751f --- /dev/null +++ b/src/main/resources/templates/fragments/body.html @@ -0,0 +1,107 @@ + + + + +
    +
    +
    +
    + + + + + + +
    + +
    +
    + + + + + + + + + + + + + + + + + + + + + +
    + + + \ No newline at end of file diff --git a/src/main/resources/templates/fragments/footer.html b/src/main/resources/templates/fragments/footer.html new file mode 100644 index 0000000..8d436d3 --- /dev/null +++ b/src/main/resources/templates/fragments/footer.html @@ -0,0 +1,29 @@ + + + + +
    + +
    + +
    + + + + © 2024 Company, Inc +
    + + +
    + +
    + + + + \ No newline at end of file diff --git a/src/main/resources/templates/fragments/header.html b/src/main/resources/templates/fragments/header.html new file mode 100644 index 0000000..3d1e75c --- /dev/null +++ b/src/main/resources/templates/fragments/header.html @@ -0,0 +1,18 @@ + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/templates/layout/basic.html b/src/main/resources/templates/layout/basic.html new file mode 100644 index 0000000..9399165 --- /dev/null +++ b/src/main/resources/templates/layout/basic.html @@ -0,0 +1,6 @@ + + + + + \ No newline at end of file diff --git a/src/main/resources/templates/user/login.html b/src/main/resources/templates/user/login.html index cf34192..d2f5fe5 100644 --- a/src/main/resources/templates/user/login.html +++ b/src/main/resources/templates/user/login.html @@ -1,15 +1,19 @@ + xmlns:sec="http://www.thymeleaf.org/extras/spring-security" + xmlns:layout="http://www.ultraq.net.nz/thymeleaf/layout" + layout:decorate="/layout/basic"> + + 로그인 - - - - + + + + @@ -61,15 +65,26 @@

    로그인

    body: JSON.stringify(data) // JSON 문자열로 변환하여 전송 }) .then(response => response.json()) - .then(data => { - if (data.message === "로그인 성공!") { - // 로그인 성공 시 리다이렉트 (예: 대시보드로 이동) - window.location.href = '/'; // 원하는 페이지로 리다이렉트 - } else { - // 로그인 실패 시 메시지 표시 - alert(data.message); // 실패 메시지 출력 - } - }) + + .then(data => { + // 서버 응답에서 메시지가 로그인 성공이라면 + if (data.message === "로그인 성공!") { + // 토큰이 존재하면 localStorage에 저장 + if (data.token) { + localStorage.setItem("token", data.token); // 토큰을 localStorage에 저장 + console.log("Token saved:", data.token); + + // 리다이렉트 처리 (예: 대시보드 페이지로 이동) + window.location.href = data.redirect || '/'; // redirect 값이 있으면 그 경로로, 없으면 기본 '/'로 리다이렉트 + } else { + console.error("Token not found in response"); + alert("토큰이 응답에 없습니다."); + } + } else { + // 로그인 실패 시 메시지 표시 + alert(data.message); // 실패 메시지 출력 + } + }) .catch(error => { console.error('로그인 실패:', error); alert('로그인 요청 중 오류가 발생했습니다.'); @@ -77,12 +92,6 @@

    로그인

    }); - - - - - - From b767de0801866c021f9dca5193a3856984d82213 Mon Sep 17 00:00:00 2001 From: KangSeongKwan Date: Mon, 10 Feb 2025 16:24:06 +0900 Subject: [PATCH 14/73] Second Version for Like --- build.gradle | 5 ++ .../cookofking/controller/LikeController.java | 58 +++++++++++++++---- .../cookofking/controller/PostController.java | 1 - .../cookofking/controller/UserController.java | 4 +- .../cookofking/repository/LikeRepository.java | 9 ++- .../service/ImagemappingService.java | 1 - .../cookofking/service/LikeService.java | 43 +++++++++++--- .../resources/templates/Post/post-view.html | 15 ++++- 8 files changed, 107 insertions(+), 29 deletions(-) diff --git a/build.gradle b/build.gradle index 752ad81..7feaefe 100644 --- a/build.gradle +++ b/build.gradle @@ -46,3 +46,8 @@ dependencies { tasks.named('test') { useJUnitPlatform() } + +tasks.withType(JavaCompile).configureEach { + options.compilerArgs.add("-parameters") +} + diff --git a/src/main/java/com/recipe/cookofking/controller/LikeController.java b/src/main/java/com/recipe/cookofking/controller/LikeController.java index 08dc0e3..0241b2f 100644 --- a/src/main/java/com/recipe/cookofking/controller/LikeController.java +++ b/src/main/java/com/recipe/cookofking/controller/LikeController.java @@ -1,16 +1,13 @@ package com.recipe.cookofking.controller; -import com.recipe.cookofking.dto.LikeDto; import com.recipe.cookofking.service.LikeService; import org.springframework.http.ResponseEntity; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RequestParam; -import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.bind.annotation.*; + +import java.util.Map; @RestController @RequestMapping("/api/v1/likes") - public class LikeController { private final LikeService likeService; @@ -18,13 +15,50 @@ public LikeController(LikeService likeService) { this.likeService = likeService; } - @GetMapping - public ResponseEntity getLikes(@RequestParam Integer postId) { + // 📌 좋아요 추가/취소 API + @PostMapping + public ResponseEntity toggleLike(@RequestBody Map request) { + try { + // 🔥 값이 `null`이거나 형 변환이 안 되는 경우 방지 + if (!request.containsKey("postId") || !request.containsKey("userId")) { + return ResponseEntity.badRequest().body(Map.of("error", "postId와 userId가 필요합니다.")); + } + + int postId; + int userId; + + try { + postId = (int) request.get("postId"); + userId = (int) request.get("userId"); + } catch (NumberFormatException e) { + return ResponseEntity.badRequest().body(Map.of("error", "postId 또는 userId 형식이 올바르지 않습니다.")); + } + + int likeCount = likeService.toggleLike(postId, userId); + return ResponseEntity.ok(Map.of("likeCount", likeCount)); + + } catch (Exception e) { + return ResponseEntity.internalServerError().body(Map.of("error", e.getMessage())); + } + } + + // 📌 좋아요 개수 조회 API + @GetMapping("/{postId}") + public ResponseEntity getLikeCount(@PathVariable(name = "postId") Integer postId) { try { - LikeDto responseDto = likeService.getLikes(postId); - return ResponseEntity.ok(responseDto); - } catch (IllegalArgumentException e) { - return ResponseEntity.badRequest().body(e.getMessage()); + Integer parsedPostId; + + try { + parsedPostId = postId; + } catch (NumberFormatException e) { + return ResponseEntity.badRequest().body(Map.of("error", "postId 형식이 올바르지 않습니다.")); + } + + int likeCount = likeService.getLikeCount(parsedPostId); + return ResponseEntity.ok(Map.of("likeCount", likeCount)); + + } catch (Exception e) { + return ResponseEntity.internalServerError().body(Map.of("error", e.getMessage())); } } } diff --git a/src/main/java/com/recipe/cookofking/controller/PostController.java b/src/main/java/com/recipe/cookofking/controller/PostController.java index 8395080..f6e7bc1 100644 --- a/src/main/java/com/recipe/cookofking/controller/PostController.java +++ b/src/main/java/com/recipe/cookofking/controller/PostController.java @@ -23,7 +23,6 @@ public String showWriteForm() { return "Post/post-write"; } - // 레시피 조회 폼 @RequestMapping("/view/{postid}") public String showViewForm(@PathVariable Integer postid, Model model) { diff --git a/src/main/java/com/recipe/cookofking/controller/UserController.java b/src/main/java/com/recipe/cookofking/controller/UserController.java index 97fee4e..a6e0e08 100644 --- a/src/main/java/com/recipe/cookofking/controller/UserController.java +++ b/src/main/java/com/recipe/cookofking/controller/UserController.java @@ -31,9 +31,7 @@ public class UserController { private final UserService userService; - - - + /* 로그인 페이지 */ @GetMapping("/login") public String login( ) { diff --git a/src/main/java/com/recipe/cookofking/repository/LikeRepository.java b/src/main/java/com/recipe/cookofking/repository/LikeRepository.java index 749c434..a29f32e 100644 --- a/src/main/java/com/recipe/cookofking/repository/LikeRepository.java +++ b/src/main/java/com/recipe/cookofking/repository/LikeRepository.java @@ -1,10 +1,17 @@ package com.recipe.cookofking.repository; import com.recipe.cookofking.entity.Like; +import com.recipe.cookofking.entity.Post; +import com.recipe.cookofking.entity.User; + +import java.util.Optional; + import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.stereotype.Repository; @Repository public interface LikeRepository extends JpaRepository { // Add custom query methods if needed - long countByPostId(Integer postId); // 특정 게시물의 좋아요 개수를 반환 + + Optional findByUserAndPost(User user, Post post); // 특정 유저가 특정 게시글을 좋아요 했는지 조회 + int countByPost(Post post); // 특정 게시글의 좋아요 개수 조회 } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/service/ImagemappingService.java b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java index db36363..a356a60 100644 --- a/src/main/java/com/recipe/cookofking/service/ImagemappingService.java +++ b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java @@ -11,7 +11,6 @@ import org.springframework.web.multipart.MultipartFile; import software.amazon.awssdk.services.s3.S3Client; import software.amazon.awssdk.services.s3.model.PutObjectRequest; - import java.io.IOException; import java.util.UUID; diff --git a/src/main/java/com/recipe/cookofking/service/LikeService.java b/src/main/java/com/recipe/cookofking/service/LikeService.java index 56d8cdc..cd7722b 100644 --- a/src/main/java/com/recipe/cookofking/service/LikeService.java +++ b/src/main/java/com/recipe/cookofking/service/LikeService.java @@ -1,27 +1,54 @@ package com.recipe.cookofking.service; -import com.recipe.cookofking.dto.LikeDto; +import com.recipe.cookofking.entity.Like; +import com.recipe.cookofking.entity.Post; +import com.recipe.cookofking.entity.User; import com.recipe.cookofking.repository.LikeRepository; import com.recipe.cookofking.repository.PostRepository; +import com.recipe.cookofking.repository.UserRepository; import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import java.util.Optional; @Service public class LikeService { - // Add Like related business logic here private final LikeRepository likeRepository; private final PostRepository postRepository; + private final UserRepository userRepository; - public LikeService(LikeRepository likeRepository, PostRepository postRepository) { + public LikeService(LikeRepository likeRepository, PostRepository postRepository, UserRepository userRepository) { this.likeRepository = likeRepository; this.postRepository = postRepository; + this.userRepository = userRepository; } - public LikeDto getLikes(Integer postId) { - if (!postRepository.existsById(postId)) { - throw new IllegalArgumentException("Post not found"); + // 좋아요 추가/취소 + @Transactional + public int toggleLike(Integer postId, Integer userId) { + User user = userRepository.findById(userId) + .orElseThrow(() -> new IllegalArgumentException("User not found")); + Post post = postRepository.findById(postId) + .orElseThrow(() -> new IllegalArgumentException("Post not found")); + + Optional existingLike = likeRepository.findByUserAndPost(user, post); + + if (existingLike.isPresent()) { + likeRepository.delete(existingLike.get()); // 이미 좋아요 했으면 취소 + } else { + Like newLike = Like.builder().user(user).post(post).build(); + likeRepository.save(newLike); // 좋아요 추가 } - long likesCount = likeRepository.countByPostId(postId); - return new LikeDto(); + return likeRepository.countByPost(post); // 📌 반환 타입 맞추기 + } + + // 좋아요 조회 + @Transactional(readOnly = true) + public int getLikeCount(Integer postId) { + Post post = postRepository.findById(postId) + .orElseThrow(() -> new IllegalArgumentException("Post not found")); + + return likeRepository.countByPost(post); // 📌 Long 변환 } } diff --git a/src/main/resources/templates/Post/post-view.html b/src/main/resources/templates/Post/post-view.html index b982e9c..ff76a4a 100644 --- a/src/main/resources/templates/Post/post-view.html +++ b/src/main/resources/templates/Post/post-view.html @@ -85,7 +85,7 @@

    레시피 제목

    - @@ -132,12 +132,16 @@

    조리 순서

    + + + + +
    + +
    + + +
    + + +
    + +
    +
    0개의 맛있는 레시피가 있습니다.
    +
    + + +
    + + + +
    +
    + +
    +
    + + +
    + + +
    + 레시피 이미지 +
    + + +
    +
    레시피 제목
    +

    내용 요약

    + + +
    + 홍길동 + 👍 0 + 👁️ 0 +
    +
    + +
    +
    +
    + + + +
    + +
    +
    + + From b2b8e6f2dbdd09293200d74c2ed443bf24784598 Mon Sep 17 00:00:00 2001 From: phc979 Date: Mon, 10 Feb 2025 17:30:40 +0900 Subject: [PATCH 16/73] fix : pagination-item-index --- src/main/resources/templates/post/posts.html | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/resources/templates/post/posts.html b/src/main/resources/templates/post/posts.html index d423223..c331eb7 100644 --- a/src/main/resources/templates/post/posts.html +++ b/src/main/resources/templates/post/posts.html @@ -201,7 +201,7 @@
    레시피 제목
    )}" th:classappend="${postPage.number == pageNum} ? 'active'"> From 1cc9c65f283570cfbb3c8d6e222a1f021062746d Mon Sep 17 00:00:00 2001 From: phc979 Date: Mon, 10 Feb 2025 19:23:53 +0900 Subject: [PATCH 17/73] feature : post write session --- .../controller/PostRestController.java | 44 ++++++++++++++++--- .../cookofking/service/PostService.java | 25 ++++++++--- .../cookofking/service/UserService.java | 9 +++- .../resources/templates/post/post-write.html | 37 +++++++++++----- src/main/resources/templates/user/login.html | 41 +++++++++++------ 5 files changed, 119 insertions(+), 37 deletions(-) diff --git a/src/main/java/com/recipe/cookofking/controller/PostRestController.java b/src/main/java/com/recipe/cookofking/controller/PostRestController.java index 6e73c1d..64235dc 100644 --- a/src/main/java/com/recipe/cookofking/controller/PostRestController.java +++ b/src/main/java/com/recipe/cookofking/controller/PostRestController.java @@ -1,12 +1,21 @@ package com.recipe.cookofking.controller; +import com.auth0.jwt.JWT; +import com.auth0.jwt.algorithms.Algorithm; +import com.auth0.jwt.interfaces.DecodedJWT; +import com.recipe.cookofking.config.jwt.JwtProperties; +import com.recipe.cookofking.dto.UserDto; import com.recipe.cookofking.dto.post.RecipeSubmissionDto; import com.recipe.cookofking.service.ImagemappingService; import com.recipe.cookofking.service.PostService; +import com.recipe.cookofking.service.UserService; import lombok.RequiredArgsConstructor; import org.springframework.http.ResponseEntity; import org.springframework.web.bind.annotation.*; +import java.util.HashMap; +import java.util.Map; + @RestController @RequestMapping("/api/v1") @RequiredArgsConstructor @@ -14,15 +23,40 @@ public class PostRestController { private final PostService postService; private final ImagemappingService imagemappingService; + private final UserService userService; + // 레시피 저장 API @PostMapping("/submit-recipe") - public ResponseEntity submitRecipe(@RequestBody RecipeSubmissionDto submissionDto) { - // 1. 이미지 검증 + public ResponseEntity> submitRecipe(@RequestBody RecipeSubmissionDto submissionDto, + @RequestHeader("Authorization") String authorizationHeader) { + // JWT 토큰에서 username 추출 + String token = authorizationHeader.replace("Bearer ", ""); + DecodedJWT decodedJWT = JWT.require(Algorithm.HMAC512(JwtProperties.SECRET)) + .build() + .verify(token); + + String username = decodedJWT.getClaim("uid").asString(); // 'uid' 클레임에서 username 추출 + + System.out.println(username); + + // UserService에서 UserDto 조회 + UserDto userDto = userService.findUserByUsername(username); + + // PostDto에 UserDto 설정 + submissionDto.getRecipeData().setUser(userDto); + + System.out.println(userDto); + + // 이미지 검증 imagemappingService.validateAndMarkPermanent(submissionDto.getValidationData()); - // 2. 레시피 저장 - postService.savePost(submissionDto.getRecipeData()); + Integer postId = postService.savePost(submissionDto.getRecipeData()); + + // postId를 포함한 응답 반환 + Map response = new HashMap<>(); + response.put("message", "레시피가 성공적으로 저장되었습니다!"); + response.put("postId", postId); - return ResponseEntity.ok("레시피가 성공적으로 저장되었습니다!"); + return ResponseEntity.ok(response); } } diff --git a/src/main/java/com/recipe/cookofking/service/PostService.java b/src/main/java/com/recipe/cookofking/service/PostService.java index 6228b1b..29f78b8 100644 --- a/src/main/java/com/recipe/cookofking/service/PostService.java +++ b/src/main/java/com/recipe/cookofking/service/PostService.java @@ -2,8 +2,10 @@ import com.recipe.cookofking.dto.post.PostDto; import com.recipe.cookofking.dto.post.PostViewDto; import com.recipe.cookofking.entity.Post; +import com.recipe.cookofking.entity.User; import com.recipe.cookofking.mapper.PostMapper; import com.recipe.cookofking.repository.PostRepository; +import com.recipe.cookofking.repository.UserRepository; import lombok.RequiredArgsConstructor; import org.springframework.data.domain.Page; import org.springframework.data.domain.Pageable; @@ -13,19 +15,30 @@ @Service @RequiredArgsConstructor public class PostService { - private final PostRepository postRepository; @Transactional - public void savePost(PostDto postDto) { - // 1. Post 엔티티 생성 + private final PostRepository postRepository; + private final UserRepository userRepository; + + @Transactional + public Integer savePost(PostDto postDto) { + // 1. DB에서 User 엔티티 조회 + User user = userRepository.findById(postDto.getUser().getId()) + .orElseThrow(() -> new IllegalArgumentException("사용자를 찾을 수 없습니다.")); + + // 2. Post 엔티티 생성 (User 엔티티 포함) Post post = Post.builder() .title(postDto.getTitle()) .content(postDto.getContent()) - .ingredients(postDto.getIngredients()) // JSON 문자열로 저장 - .instructions(postDto.getInstructions()) // JSON 문자열로 저장 + .ingredients(postDto.getIngredients()) + .instructions(postDto.getInstructions()) .mainImageS3URL(postDto.getMainImageS3URL()) + .user(user) // 영속 상태의 User 엔티티 설정 .build(); - // 2. 레시피 저장 + // 3. 레시피 저장 Post savedPost = postRepository.save(post); + + // 4. 저장된 게시글의 ID 반환 + return savedPost.getId(); } diff --git a/src/main/java/com/recipe/cookofking/service/UserService.java b/src/main/java/com/recipe/cookofking/service/UserService.java index 2328b6d..f0da978 100644 --- a/src/main/java/com/recipe/cookofking/service/UserService.java +++ b/src/main/java/com/recipe/cookofking/service/UserService.java @@ -4,6 +4,7 @@ import org.springframework.security.authentication.AuthenticationManager; import org.springframework.security.authentication.UsernamePasswordAuthenticationToken; import org.springframework.security.core.Authentication; +import org.springframework.security.core.userdetails.UsernameNotFoundException; import org.springframework.security.crypto.password.PasswordEncoder; import org.springframework.stereotype.Service; @@ -67,8 +68,12 @@ public void registerUser(UserDto userDto) { userRepository.save(user); } - - + + public UserDto findUserByUsername(String username) { + User user = userRepository.findByUsername(username) + .orElseThrow(() -> new UsernameNotFoundException("사용자를 찾을 수 없습니다.")); + return UserDto.fromEntity(user); + } } \ No newline at end of file diff --git a/src/main/resources/templates/post/post-write.html b/src/main/resources/templates/post/post-write.html index 963d67e..26a18a6 100644 --- a/src/main/resources/templates/post/post-write.html +++ b/src/main/resources/templates/post/post-write.html @@ -205,18 +205,14 @@

    요리 순서

    title: $('#recipeTitle').val(), content: $('#recipeDescription').val(), mainImageS3URL: $('#mainImageUrl').val(), - - // 재료 정보 수집 후 문자열로 변환 - ingredients: JSON.stringify(collectIngredients()), - - // 요리 순서 수집 후 문자열로 변환 - instructions: JSON.stringify(collectInstructions()) + ingredients: JSON.stringify(collectIngredients()), // 재료 정보 + instructions: JSON.stringify(collectInstructions()) // 요리 순서 }; const validationData = { mainImageId: $('#mainImageId').val(), mainImageUrl: $('#mainImageUrl').val(), - stepImages: collectStepImageValidation() + stepImages: collectStepImageValidation() // 이미지 검증 데이터 }; const finalData = { @@ -224,15 +220,36 @@

    요리 순서

    validationData: validationData }; - // JSON 데이터 전송 + // JWT 토큰 로컬 스토리지에서 꺼내오기 + const token = localStorage.getItem('token'); + + if (!token) { + alert('로그인 후 다시 시도해주세요.'); + window.location.href = '/user/login'; // 로그인 페이지로 리다이렉트 + return; + } + + // JSON 데이터 전송 (Authorization 헤더 포함) $.ajax({ url: '/api/v1/submit-recipe', type: 'POST', contentType: 'application/json', data: JSON.stringify(finalData), + headers: { + 'Authorization': 'Bearer ' + token // JWT 토큰 추가 + }, success: function (response) { - alert(response); - window.location.href = '/recipes'; // 저장 후 리다이렉션 + if (response.postId) { + // postId로 상세 페이지 이동 + window.location.href = '/post/view/' + response.postId; + } else { + alert('레시피 저장은 성공했지만, 게시글 ID를 받지 못했습니다.'); + window.location.href = '/post/list'; // 예외 처리: 목록으로 이동 + } + }, + error: function (xhr, status, error) { + console.error('레시피 저장 중 오류 발생:', error); + alert('레시피 저장에 실패했습니다.'); } }); }); diff --git a/src/main/resources/templates/user/login.html b/src/main/resources/templates/user/login.html index cf34192..9707ba9 100644 --- a/src/main/resources/templates/user/login.html +++ b/src/main/resources/templates/user/login.html @@ -60,20 +60,33 @@

    로그인

    }, body: JSON.stringify(data) // JSON 문자열로 변환하여 전송 }) - .then(response => response.json()) - .then(data => { - if (data.message === "로그인 성공!") { - // 로그인 성공 시 리다이렉트 (예: 대시보드로 이동) - window.location.href = '/'; // 원하는 페이지로 리다이렉트 - } else { - // 로그인 실패 시 메시지 표시 - alert(data.message); // 실패 메시지 출력 - } - }) - .catch(error => { - console.error('로그인 실패:', error); - alert('로그인 요청 중 오류가 발생했습니다.'); - }); + .then(response => { + if (!response.ok) { + throw new Error('로그인 실패'); + } + + // 1. Authorization 헤더에서 토큰 추출 + const authHeader = response.headers.get('Authorization'); // 'Authorization' 헤더 가져오기 + const token = authHeader ? authHeader.replace('Bearer ', '') : null; // 'Bearer ' 제거 후 토큰만 추출 + + // 2. 본문 파싱 + return response.json().then(data => ({ data, token })); + }) + .then(({ data, token }) => { + console.log('응답 데이터:', data); + console.log('추출한 토큰:', token); + + if (data.message === "로그인 성공!" && token) { + localStorage.setItem('token', token); // 토큰 저장 + window.location.href = '/'; // 리다이렉트 + } else { + alert('로그인 실패: 토큰이 없습니다.'); + } + }) + .catch(error => { + console.error('로그인 요청 중 오류 발생:', error); + alert('로그인 요청 중 오류가 발생했습니다.'); + }); }); From 42fb2b3d4965daad5a693aed748d81d2eb7eded0 Mon Sep 17 00:00:00 2001 From: phc979 Date: Tue, 11 Feb 2025 00:21:42 +0900 Subject: [PATCH 18/73] feature : update-recipe & image orphan delete & jwt token ( with cookie ) --- .../cookofking/config/SecurityConfig.java | 26 +- .../config/jwt/JwtAuthorizationFilter.java | 105 +++-- .../cookofking/controller/PostController.java | 41 +- .../controller/PostRestController.java | 116 ++++- .../dto/image/ImageValidationDto.java | 9 +- .../dto/post/RecipeSubmissionDto.java | 2 + .../cookofking/entity/Imagemapping.java | 8 + .../com/recipe/cookofking/entity/Post.java | 16 + .../repository/ImagemappingRepository.java | 1 + .../service/ImagemappingService.java | 116 ++++- .../cookofking/service/PostService.java | 48 ++ .../resources/templates/post/post-edit.html | 425 ++++++++++++++++++ .../resources/templates/post/post-view.html | 14 +- .../resources/templates/post/post-write.html | 20 +- src/main/resources/templates/user/login.html | 8 +- 15 files changed, 864 insertions(+), 91 deletions(-) create mode 100644 src/main/resources/templates/post/post-edit.html diff --git a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java index b93e90a..d62fdae 100644 --- a/src/main/java/com/recipe/cookofking/config/SecurityConfig.java +++ b/src/main/java/com/recipe/cookofking/config/SecurityConfig.java @@ -60,13 +60,25 @@ public SecurityFilterChain filterChain(HttpSecurity http, AuthenticationManager .formLogin(AbstractHttpConfigurer::disable) .httpBasic(AbstractHttpConfigurer::disable) - - // 경로별 권한 설정 - .authorizeHttpRequests(auth -> auth - .requestMatchers("/api/**","/**", "/login","/register","/js/**", "/css/**", "/images/**", "/static/**").permitAll() // 로그인 경로는 인증 없이 접근 가능 - //.requestMatchers("/api/mypage").authenticated() // 인증필요시 - //.requestMatchers("/admin/**").hasRole("ADMIN") // 예시: admin 권한이 필요한 경로 - .anyRequest().authenticated()); // 다른 모든 요청은 인증 필요 + + // 경로별 권한 설정 + .authorizeHttpRequests(auth -> auth + .requestMatchers( + "/user/login", + "/user/register", + "/js/**", + "/css/**", + "/images/**", + "/static/**", + "/post/list", + "/post/view/**" + ).permitAll() // 로그인, 회원가입, 정적 리소스, 게시글 목록/보기는 인증 없이 접근 가능 + + .requestMatchers("/post/edit/**", "/api/**").authenticated() // 게시글 수정과 API 경로는 인증 필요 + + .anyRequest().permitAll() // 그 외 나머지 요청은 인증 없이 허용 + ); + return http.build(); } diff --git a/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java index b21e0e6..f72b28b 100644 --- a/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java +++ b/src/main/java/com/recipe/cookofking/config/jwt/JwtAuthorizationFilter.java @@ -3,6 +3,8 @@ import java.io.IOException; + +import jakarta.servlet.http.Cookie; import org.springframework.security.authentication.AuthenticationManager; import org.springframework.security.authentication.UsernamePasswordAuthenticationToken; import org.springframework.security.core.Authentication; @@ -38,42 +40,85 @@ public JwtAuthorizationFilter(AuthenticationManager authenticationManager, UserR @Override protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain chain) throws IOException, ServletException { - - // 헤더에서 Authorization 키 확인 + // 1. Authorization 헤더에서 토큰 추출 (API 요청용) String header = request.getHeader("Authorization"); - - if (header == null || !header.startsWith("Bearer ")) { - chain.doFilter(request, response); - return; + String token = null; + + if (header != null && header.startsWith("Bearer ")) { + token = header.replace("Bearer ", ""); + } else { + // 2. Authorization 헤더가 없으면 쿠키에서 토큰 추출 (페이지 요청용) + if (request.getCookies() != null) { + for (Cookie cookie : request.getCookies()) { + if ("token".equals(cookie.getName())) { + token = cookie.getValue(); + } + } + } } - // Authorization 헤더에서 토큰 추출 - String token = header.replace("Bearer ", ""); - - try { - String username = JWT.require(Algorithm.HMAC512(JwtProperties.SECRET)) - .build() - .verify(token) - .getClaim("username") - .asString(); - - if (username != null) { - User user = userRepository.findByUsername(username) - .orElseThrow(() -> new RuntimeException("User not found")); - PrincipalDetails principalDetails = new PrincipalDetails(user); - - Authentication authentication = new UsernamePasswordAuthenticationToken( - principalDetails, null, principalDetails.getAuthorities()); - - SecurityContextHolder.getContext().setAuthentication(authentication); + + + if (token != null) { + try { + String username = JWT.require(Algorithm.HMAC512(JwtProperties.SECRET)) + .build() + .verify(token) + .getClaim("uid") + .asString(); + + if (username != null) { + User user = userRepository.findByUsername(username) + .orElseThrow(() -> new RuntimeException("User not found")); + + PrincipalDetails principalDetails = new PrincipalDetails(user); + Authentication authentication = new UsernamePasswordAuthenticationToken( + principalDetails, null, principalDetails.getAuthorities()); + + SecurityContextHolder.getContext().setAuthentication(authentication); + } + } catch (Exception e) { + log.error("JWT 처리 중 오류 발생", e); + response.setStatus(HttpServletResponse.SC_FORBIDDEN); + return; } - } catch (Exception e) { - log.error("JWT 처리 중 오류 발생", e); - response.setStatus(HttpServletResponse.SC_FORBIDDEN); - response.getWriter().write("Forbidden: Invalid or expired token"); - return; } chain.doFilter(request, response); +// // 헤더에서 Authorization 키 확인 +// String header = request.getHeader("Authorization"); +// +// if (header == null || !header.startsWith("Bearer ")) { +// chain.doFilter(request, response); +// return; +// } +// // Authorization 헤더에서 토큰 추출 +// String token = header.replace("Bearer ", ""); +// +// try { +// String username = JWT.require(Algorithm.HMAC512(JwtProperties.SECRET)) +// .build() +// .verify(token) +// .getClaim("uid") +// .asString(); +// +// if (username != null) { +// User user = userRepository.findByUsername(username) +// .orElseThrow(() -> new RuntimeException("User not found")); +// PrincipalDetails principalDetails = new PrincipalDetails(user); +// +// Authentication authentication = new UsernamePasswordAuthenticationToken( +// principalDetails, null, principalDetails.getAuthorities()); +// +// SecurityContextHolder.getContext().setAuthentication(authentication); +// } +// } catch (Exception e) { +// log.error("JWT 처리 중 오류 발생", e); +// response.setStatus(HttpServletResponse.SC_FORBIDDEN); +// response.getWriter().write("Forbidden: Invalid or expired token"); +// return; +// } +// +// chain.doFilter(request, response); } diff --git a/src/main/java/com/recipe/cookofking/controller/PostController.java b/src/main/java/com/recipe/cookofking/controller/PostController.java index 3f02372..d6c432c 100644 --- a/src/main/java/com/recipe/cookofking/controller/PostController.java +++ b/src/main/java/com/recipe/cookofking/controller/PostController.java @@ -1,5 +1,6 @@ package com.recipe.cookofking.controller; +import com.recipe.cookofking.config.auth.PrincipalDetails; import com.recipe.cookofking.dto.post.PostDto; import com.recipe.cookofking.dto.post.PostViewDto; import com.recipe.cookofking.entity.Post; @@ -8,6 +9,9 @@ import org.springframework.data.domain.Page; import org.springframework.data.domain.PageRequest; import org.springframework.data.domain.Sort; +import org.springframework.security.access.AccessDeniedException; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.GetMapping; @@ -39,11 +43,46 @@ public String showWriteForm() { @RequestMapping("/view/{postid}") public String showViewForm(@PathVariable Integer postid, Model model) { PostViewDto postViewDto = postService.getPostById(postid); // postid로 게시글 데이터 가져오기 - model.addAttribute("post", postViewDto); // 모델에 데이터 추가 + model.addAttribute("post", postViewDto); // 게시글 정보 추가 + + // 현재 로그인한 사용자 정보 가져오기 + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + boolean isPostOwner = false; + + if (authentication != null && authentication.isAuthenticated() && + authentication.getPrincipal() instanceof PrincipalDetails) { + + PrincipalDetails principalDetails = (PrincipalDetails) authentication.getPrincipal(); + String currentUsername = principalDetails.getUsername(); + + // 작성자와 현재 로그인한 사용자가 같은지 확인 + if (postViewDto.getUsername() != null && postViewDto.getUsername().equals(currentUsername)) { + isPostOwner = true; + } + } + + model.addAttribute("isPostOwner", isPostOwner); // 작성자 여부 추가 + return "post/post-view"; // post-view.html 렌더링 } + @GetMapping("/edit/{postId}") + public String update(@PathVariable Integer postId, Model model) { + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + PrincipalDetails principalDetails = (PrincipalDetails) authentication.getPrincipal(); + String currentUsername = principalDetails.getUsername(); + + // 작성자 검증 + postService.validatePostOwner(postId, currentUsername); + + // 게시글 정보 렌더링 + PostViewDto postViewDto = postService.getPostById(postId); + model.addAttribute("post", postViewDto); + + return "Post/post-edit"; + } + @GetMapping("/list") diff --git a/src/main/java/com/recipe/cookofking/controller/PostRestController.java b/src/main/java/com/recipe/cookofking/controller/PostRestController.java index 64235dc..6ab662b 100644 --- a/src/main/java/com/recipe/cookofking/controller/PostRestController.java +++ b/src/main/java/com/recipe/cookofking/controller/PostRestController.java @@ -5,15 +5,20 @@ import com.auth0.jwt.interfaces.DecodedJWT; import com.recipe.cookofking.config.jwt.JwtProperties; import com.recipe.cookofking.dto.UserDto; +import com.recipe.cookofking.dto.image.ImageValidationDto; import com.recipe.cookofking.dto.post.RecipeSubmissionDto; import com.recipe.cookofking.service.ImagemappingService; import com.recipe.cookofking.service.PostService; import com.recipe.cookofking.service.UserService; import lombok.RequiredArgsConstructor; import org.springframework.http.ResponseEntity; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; import org.springframework.web.bind.annotation.*; +import java.util.ArrayList; import java.util.HashMap; +import java.util.List; import java.util.Map; @RestController @@ -27,36 +32,115 @@ public class PostRestController { // 레시피 저장 API @PostMapping("/submit-recipe") - public ResponseEntity> submitRecipe(@RequestBody RecipeSubmissionDto submissionDto, - @RequestHeader("Authorization") String authorizationHeader) { + public ResponseEntity> submitRecipe(@RequestBody RecipeSubmissionDto submissionDto) { // JWT 토큰에서 username 추출 - String token = authorizationHeader.replace("Bearer ", ""); - DecodedJWT decodedJWT = JWT.require(Algorithm.HMAC512(JwtProperties.SECRET)) - .build() - .verify(token); - - String username = decodedJWT.getClaim("uid").asString(); // 'uid' 클레임에서 username 추출 - - System.out.println(username); + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + String username = authentication.getName(); // 인증된 사용자 이름 // UserService에서 UserDto 조회 UserDto userDto = userService.findUserByUsername(username); - - // PostDto에 UserDto 설정 submissionDto.getRecipeData().setUser(userDto); - System.out.println(userDto); - - // 이미지 검증 + // 이미지 검증 및 영구 저장 처리 imagemappingService.validateAndMarkPermanent(submissionDto.getValidationData()); + // 레시피 저장 Integer postId = postService.savePost(submissionDto.getRecipeData()); - // postId를 포함한 응답 반환 + // 메인 이미지 및 요리 순서 이미지의 postId 갱신 + List allImageUrls = collectAllImageUrls(submissionDto.getValidationData()); + imagemappingService.updateImagePostId(allImageUrls, postId); + + // 응답 반환 Map response = new HashMap<>(); response.put("message", "레시피가 성공적으로 저장되었습니다!"); response.put("postId", postId); return ResponseEntity.ok(response); } + + @PostMapping("/update-recipe") + public ResponseEntity> updateRecipe(@RequestBody RecipeSubmissionDto submissionDto) { + // JWT 토큰에서 사용자 정보 추출 + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + String username = authentication.getName(); // 인증된 사용자 이름 + +// String token = authorizationHeader.replace("Bearer ", ""); +// DecodedJWT decodedJWT = JWT.require(Algorithm.HMAC512(JwtProperties.SECRET)).build().verify(token); +// String username = decodedJWT.getClaim("uid").asString(); + + // 사용자 정보 조회 + UserDto userDto = userService.findUserByUsername(username); + submissionDto.getRecipeData().setUser(userDto); + + // 게시글 ID 확인 + Integer postId = submissionDto.getRecipeData().getId(); + if (postId == null) { + Map errorResponse = new HashMap<>(); + errorResponse.put("message", "게시글 ID가 누락되었습니다. 수정 요청에는 게시글 ID가 필요합니다."); + return ResponseEntity.badRequest().body(errorResponse); + } + + // 게시글 소유자 검증 + if (!postService.isUserOwnerOfPost(postId, userDto.getId())) { + Map errorResponse = new HashMap<>(); + errorResponse.put("message", "해당 게시글에 대한 수정 권한이 없습니다."); + return ResponseEntity.status(403).body(errorResponse); // 403 Forbidden + } + + // 고아 이미지 소유자 검증 + if (!validateOrphanedImageOwnership(submissionDto.getValidationData(), postId)) { + Map errorResponse = new HashMap<>(); + errorResponse.put("message", "다른 게시글에 속한 이미지를 삭제할 수 없습니다."); + return ResponseEntity.status(403).body(errorResponse); // 403 Forbidden + } + + // 이미지 검증 및 영구 저장 처리 + imagemappingService.validateAndMarkPermanent(submissionDto.getValidationData()); + + // 게시글 수정 + Integer updatedPostId = postService.updatePost(submissionDto.getRecipeData()); + + // 새로 추가된 이미지들의 postId 갱신 + List allImageUrls = collectAllImageUrls(submissionDto.getValidationData()); + imagemappingService.updateImagePostId(allImageUrls, updatedPostId); + + // 성공 응답 반환 + Map response = new HashMap<>(); + response.put("message", "레시피가 성공적으로 수정되었습니다!"); + response.put("postId", updatedPostId); + + return ResponseEntity.ok(response); + } + + + private List collectAllImageUrls(ImageValidationDto validationData) { + List imageUrls = new ArrayList<>(); + + // 메인 이미지 URL 추가 + if (validationData.getMainImage() != null && validationData.getMainImage().getImageUrl() != null) { + imageUrls.add(validationData.getMainImage().getImageUrl()); + } + + // 요리 순서 이미지 URL 추가 + validationData.getStepImages().forEach(stepImage -> { + if (stepImage.getImageUrl() != null) { + imageUrls.add(stepImage.getImageUrl()); + } + }); + + return imageUrls; + } + + private boolean validateOrphanedImageOwnership(ImageValidationDto validationData, Integer postId) { + for (String orphanedUrl : validationData.getOrphanedUrls()) { + boolean isOrphanedImageLinked = imagemappingService.isImageLinkedToPost(orphanedUrl, postId); + if (!isOrphanedImageLinked) { + return false; // 고아 이미지가 현재 게시글에 속하지 않음 + } + } + return true; // 모든 고아 이미지가 현재 게시글에 속함 + } + + } diff --git a/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java b/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java index 12732d0..c6353de 100644 --- a/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java +++ b/src/main/java/com/recipe/cookofking/dto/image/ImageValidationDto.java @@ -5,6 +5,7 @@ import lombok.Data; import lombok.NoArgsConstructor; +import java.util.ArrayList; import java.util.List; @Data @@ -12,12 +13,12 @@ @NoArgsConstructor @Builder public class ImageValidationDto { - private Integer mainImageId; - private String mainImageUrl; - private List stepImages; + private ImageInfoDto mainImage = new ImageInfoDto(); // 기본값 추가 + private List stepImages = new ArrayList<>(); + private List orphanedUrls = new ArrayList<>(); @Data - public static class StepImageDto { + public static class ImageInfoDto { private Integer imageId; private String imageUrl; } diff --git a/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java b/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java index 6289c2d..55c4394 100644 --- a/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java +++ b/src/main/java/com/recipe/cookofking/dto/post/RecipeSubmissionDto.java @@ -6,6 +6,8 @@ import lombok.Data; import lombok.NoArgsConstructor; +import java.util.List; + @Data @AllArgsConstructor @NoArgsConstructor diff --git a/src/main/java/com/recipe/cookofking/entity/Imagemapping.java b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java index b2ab25d..bb2c490 100644 --- a/src/main/java/com/recipe/cookofking/entity/Imagemapping.java +++ b/src/main/java/com/recipe/cookofking/entity/Imagemapping.java @@ -39,4 +39,12 @@ public class Imagemapping { public void markAsPermanent() { this.isTemp = false; } + public void unmarkAsPermanent() { + this.isTemp = true; + } + + public void setPost(Post post) { + this.post = post; + } + } diff --git a/src/main/java/com/recipe/cookofking/entity/Post.java b/src/main/java/com/recipe/cookofking/entity/Post.java index 299b991..9338a9a 100644 --- a/src/main/java/com/recipe/cookofking/entity/Post.java +++ b/src/main/java/com/recipe/cookofking/entity/Post.java @@ -1,5 +1,6 @@ package com.recipe.cookofking.entity; +import com.recipe.cookofking.dto.post.PostDto; import jakarta.persistence.*; import lombok.*; import org.springframework.data.annotation.CreatedDate; @@ -80,4 +81,19 @@ public void decrementLikeCount() { public void incrementLikeCount() { this.likeCount += 1; } + + public Post updateFromDto(PostDto postDto) { + return Post.builder() + .id(this.id) // 기존 ID 유지 + .user(this.user) // 기존 작성자 유지 + .title(postDto.getTitle()) + .content(postDto.getContent()) + .ingredients(postDto.getIngredients()) + .instructions(postDto.getInstructions()) + .mainImageS3URL(postDto.getMainImageS3URL()) + .viewCount(this.viewCount) // 조회수 유지 + .likeCount(this.likeCount) // 좋아요 수 유지 + .createdDate(this.createdDate) // 생성일 유지 + .build(); + } } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java b/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java index 89e38f7..c58ef76 100644 --- a/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java +++ b/src/main/java/com/recipe/cookofking/repository/ImagemappingRepository.java @@ -8,4 +8,5 @@ @Repository public interface ImagemappingRepository extends JpaRepository { Optional findByIdAndS3Url(Integer id, String s3Url); + Optional findByS3Url(String s3Url); } \ No newline at end of file diff --git a/src/main/java/com/recipe/cookofking/service/ImagemappingService.java b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java index db36363..42962b1 100644 --- a/src/main/java/com/recipe/cookofking/service/ImagemappingService.java +++ b/src/main/java/com/recipe/cookofking/service/ImagemappingService.java @@ -3,16 +3,20 @@ import com.recipe.cookofking.dto.image.ImageValidationDto; import com.recipe.cookofking.dto.image.ImagemappingDto; import com.recipe.cookofking.entity.Imagemapping; +import com.recipe.cookofking.entity.Post; import com.recipe.cookofking.mapper.ImagemappingMapper; import com.recipe.cookofking.repository.ImagemappingRepository; +import com.recipe.cookofking.repository.PostRepository; import lombok.RequiredArgsConstructor; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; import org.springframework.web.multipart.MultipartFile; import software.amazon.awssdk.services.s3.S3Client; import software.amazon.awssdk.services.s3.model.PutObjectRequest; import java.io.IOException; +import java.util.List; import java.util.UUID; @Service @@ -21,11 +25,14 @@ public class ImagemappingService { private final S3Client s3Client; private final ImagemappingRepository imagemappingRepository; + private final PostRepository postRepository; // application.properties에서 버킷 이름 주입 @Value("${cloud.aws.s3.bucket}") private String bucketName; + + @Transactional public ImagemappingDto uploadImage(MultipartFile file) throws IOException { // S3에 업로드할 파일명 생성 String fileName = "recipes/" + UUID.randomUUID() + "_" + file.getOriginalFilename(); @@ -54,23 +61,110 @@ public ImagemappingDto uploadImage(MultipartFile file) throws IOException { // 저장된 이미지 정보를 DTO로 변환하여 반환 return ImagemappingMapper.toDto(savedImage); } - + @Transactional public void validateAndMarkPermanent(ImageValidationDto validationDto) { - // 메인 이미지 검증 - imagemappingRepository.findByIdAndS3Url(validationDto.getMainImageId(), validationDto.getMainImageUrl()) - .ifPresent(image -> { - image.markAsPermanent(); - imagemappingRepository.save(image); - }); + System.out.println("Starting image validation and marking process..."); - // 조리 순서 이미지 검증 - validationDto.getStepImages().forEach(stepImage -> { - imagemappingRepository.findByIdAndS3Url(stepImage.getImageId(), stepImage.getImageUrl()) - .ifPresent(image -> { + // 1. 메인 이미지 검증 및 영구 저장 + if (validationDto.getMainImage() != null && validationDto.getMainImage().getImageId() != null) { + System.out.println("Validating main image: " + validationDto.getMainImage().getImageUrl()); + imagemappingRepository.findByIdAndS3Url(validationDto.getMainImage().getImageId(), validationDto.getMainImage().getImageUrl()) + .ifPresentOrElse(image -> { image.markAsPermanent(); imagemappingRepository.save(image); + System.out.println("Main image marked as permanent."); + }, () -> { + System.out.println("Main image not found for marking."); }); + } + + // 2. 조리 순서 이미지 검증 및 영구 저장 + validationDto.getStepImages().forEach(stepImage -> { + if (stepImage.getImageId() != null) { + System.out.println("Validating step image with ID: " + stepImage.getImageId() + ", URL: " + stepImage.getImageUrl()); + imagemappingRepository.findByIdAndS3Url(stepImage.getImageId(), stepImage.getImageUrl()) + .ifPresentOrElse(image -> { + image.markAsPermanent(); + imagemappingRepository.save(image); + System.out.println("Step image marked as permanent."); + }, () -> { + System.out.println("Step image not found for marking."); + }); + } }); + + // 3. 고아 이미지 처리 (영구 저장 해제 또는 삭제) + if (validationDto.getOrphanedUrls() != null && !validationDto.getOrphanedUrls().isEmpty()) { + System.out.println("Processing orphaned images: " + validationDto.getOrphanedUrls()); + + validationDto.getOrphanedUrls().forEach(orphanUrl -> { + System.out.println("Checking orphaned image with URL: " + orphanUrl); + imagemappingRepository.findByS3Url(orphanUrl) + .ifPresentOrElse(orphanedImage -> { + orphanedImage.unmarkAsPermanent(); // 영구 저장 해제 + imagemappingRepository.save(orphanedImage); + System.out.println("Orphaned image unmarked as permanent."); + }, () -> { + System.out.println("Orphaned image not found in repository."); + }); + }); + } else { + System.out.println("No orphaned images to process."); + } + + System.out.println("Image validation and marking process completed."); + } + + + public boolean isImageLinkedToPost(String imageUrl, Integer postId) { + return imagemappingRepository.findByS3Url(imageUrl) + .map(image -> image.getPost().getId().equals(postId)) // 이미지의 postId와 요청된 postId 비교 + .orElse(false); // 이미지가 존재하지 않으면 false 반환 + } + + @Transactional + public void updateImagePostId(List imageUrls, Integer postId) { + System.out.println("Starting updateImagePostId process..."); + System.out.println("Target Post ID: " + postId); + System.out.println("Image URLs to update: " + imageUrls); + + // Post 객체 조회 + Post post = postRepository.findById(postId) + .orElseThrow(() -> { + System.out.println("Post not found with ID: " + postId); + return new RuntimeException("해당 게시글을 찾을 수 없습니다."); + }); + + System.out.println("Post found: " + post.getTitle()); + + // 이미지에 Post 설정 + for (String imageUrl : imageUrls) { + System.out.println("Processing image with URL: " + imageUrl); + + imagemappingRepository.findByS3Url(imageUrl) + .ifPresentOrElse(image -> { + System.out.println("Image found in repository. Updating post association."); + image.setPost(post); // Post 객체 설정 + imagemappingRepository.save(image); + System.out.println("Image successfully updated with Post ID: " + post.getId()); + }, () -> { + System.out.println("Image not found in repository for URL: " + imageUrl); + }); + } + + System.out.println("updateImagePostId process completed."); +// // Post 객체 조회 +// Post post = postRepository.findById(postId) +// .orElseThrow(() -> new RuntimeException("해당 게시글을 찾을 수 없습니다.")); +// +// // 이미지에 Post 설정 +// for (String imageUrl : imageUrls) { +// imagemappingRepository.findByS3Url(imageUrl) +// .ifPresent(image -> { +// image.setPost(post); // Post 객체 설정 +// imagemappingRepository.save(image); +// }); +// } } diff --git a/src/main/java/com/recipe/cookofking/service/PostService.java b/src/main/java/com/recipe/cookofking/service/PostService.java index 29f78b8..51684f5 100644 --- a/src/main/java/com/recipe/cookofking/service/PostService.java +++ b/src/main/java/com/recipe/cookofking/service/PostService.java @@ -9,9 +9,12 @@ import lombok.RequiredArgsConstructor; import org.springframework.data.domain.Page; import org.springframework.data.domain.Pageable; +import org.springframework.security.access.AccessDeniedException; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; +import java.time.LocalDateTime; + @Service @RequiredArgsConstructor public class PostService { @@ -41,6 +44,29 @@ public Integer savePost(PostDto postDto) { return savedPost.getId(); } + @Transactional(readOnly = true) + public boolean isUserOwnerOfPost(Integer postId, Integer userId) { + return postRepository.findById(postId) + .map(post -> post.getUser().getId().equals(userId)) + .orElse(false); // 게시글이 존재하지 않으면 false 반환 + } + + + + @Transactional + public Integer updatePost(PostDto postDto) { + // 기존 게시글 조회 + Post existingPost = postRepository.findById(postDto.getId()) + .orElseThrow(() -> new IllegalArgumentException("해당 게시글을 찾을 수 없습니다. ID: " + postDto.getId())); + + // 정적 팩토리 메서드로 새로운 엔티티 생성 + Post updatedPost = existingPost.updateFromDto(postDto); + + // 수정된 엔티티 저장 + postRepository.save(updatedPost); + + return updatedPost.getId(); + } @Transactional @@ -74,4 +100,26 @@ public Page getPostList(Pageable pageable) { return postRepository.findAll(pageable).map(PostMapper::toDto); } + + public void validatePostOwner(Integer postId, String currentUsername) { + Post post = postRepository.findById(postId) + .orElseThrow(() -> new RuntimeException("Post not found with id: " + postId)); + + // 1. 로그인 여부 확인 + if (currentUsername == null || currentUsername.isEmpty()) { + throw new AccessDeniedException("로그인 후 수정 가능합니다."); + } + + // 2. 탈퇴한 회원 여부 확인 (post.getUser()가 null인 경우) + if (post.getUser() == null) { + throw new AccessDeniedException("탈퇴한 회원의 게시글은 수정할 수 없습니다."); + } + + // 3. 게시글 소유자 확인 + if (!post.getUser().getUsername().equals(currentUsername)) { + throw new AccessDeniedException("해당 게시글에 대한 수정 권한이 없습니다."); + } + } + + } diff --git a/src/main/resources/templates/post/post-edit.html b/src/main/resources/templates/post/post-edit.html new file mode 100644 index 0000000..868e271 --- /dev/null +++ b/src/main/resources/templates/post/post-edit.html @@ -0,0 +1,425 @@ + + + + + + 레시피 등록 + + + + + + + +
    +
    +
    + + +
    +
    +

    레시피 등록

    + +
    + + +
    + + +
    + + +
    + +
    + + +
    +
    + +
    + + +

    재료 정보

    +

    재료가 남거나 부족하지 않도록 정확한 계량정보를 적어주세요.

    + +
    +
    +
    + +
    +
    + +
    +
    + +
    +
    + +
    +
    +
    + + + +
    + + +

    요리 순서

    +

    + 요리의 맛이 좌우될 수 있는 중요한 부분은 빠짐없이 적어주세요.
    + 예) 10분간 약한불로 익혀주세요. 마늘편은 충분히 익혀 매운 맛을 제거하세요. +

    + +
    + +
    + + + +
    + +
    +
    + +
    +
    +
    + + + + + diff --git a/src/main/resources/templates/post/post-view.html b/src/main/resources/templates/post/post-view.html index 41e88dc..e8a5377 100644 --- a/src/main/resources/templates/post/post-view.html +++ b/src/main/resources/templates/post/post-view.html @@ -86,7 +86,10 @@

    레시피 제목

    조회수: 0
    - + +
    @@ -134,6 +137,13 @@

    조리 순서

    -
    - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/src/main/resources/templates/fragments/footer.html b/src/main/resources/templates/fragments/footer.html deleted file mode 100644 index 8d436d3..0000000 --- a/src/main/resources/templates/fragments/footer.html +++ /dev/null @@ -1,29 +0,0 @@ - - - - -
    - -
    - -
    - - - - © 2024 Company, Inc -
    - - -
    - -
    - - - - \ No newline at end of file diff --git a/src/main/resources/templates/fragments/header.html b/src/main/resources/templates/fragments/header.html deleted file mode 100644 index 3d1e75c..0000000 --- a/src/main/resources/templates/fragments/header.html +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/src/main/resources/templates/index.html b/src/main/resources/templates/index.html index a5fce8a..3b9067e 100644 --- a/src/main/resources/templates/index.html +++ b/src/main/resources/templates/index.html @@ -1,12 +1,22 @@ - - + - -Insert title here + +Index page + + + -메인페이지 + + + +
    Header
    +
    Footer
    + - \ No newline at end of file + + + diff --git a/src/main/resources/templates/layout/basic.html b/src/main/resources/templates/layout/basic.html deleted file mode 100644 index 9399165..0000000 --- a/src/main/resources/templates/layout/basic.html +++ /dev/null @@ -1,6 +0,0 @@ - - - - - \ No newline at end of file diff --git a/src/main/resources/templates/layout/footer.html b/src/main/resources/templates/layout/footer.html new file mode 100644 index 0000000..1e9929c --- /dev/null +++ b/src/main/resources/templates/layout/footer.html @@ -0,0 +1,21 @@ + + + + +Footer + + + +
    +
    +
    +

    © 2025 Company, Inc. All rights reserved.

    +
    +
    +
    + +
    + + + \ No newline at end of file diff --git a/src/main/resources/templates/layout/header.html b/src/main/resources/templates/layout/header.html new file mode 100644 index 0000000..279479c --- /dev/null +++ b/src/main/resources/templates/layout/header.html @@ -0,0 +1,40 @@ + + + + + Title + + + +
    +
    +
    + + +
    + + + + 로그인 + 회원가입 + + + + +
    +
    +
    + +
    +
    + + + + + + diff --git a/src/main/resources/templates/user/login.html b/src/main/resources/templates/user/login.html index d2f5fe5..66d27d9 100644 --- a/src/main/resources/templates/user/login.html +++ b/src/main/resources/templates/user/login.html @@ -1,97 +1,82 @@ - - - - - + + - 로그인 - - + + My Page + + + + - - - -