From 267d1b3921f919e31c883ad6abb840225ae72502 Mon Sep 17 00:00:00 2001 From: Jungwoo Kim Date: Thu, 1 Feb 2024 21:25:26 +0900 Subject: [PATCH 1/2] =?UTF-8?q?fix:=20LocalDateTime=20=ED=83=80=EC=9E=85?= =?UTF-8?q?=20=EC=A7=81=EB=A0=AC=ED=99=94/=EC=97=AD=EC=A7=81=EB=A0=AC?= =?UTF-8?q?=ED=99=94=20=EC=98=A4=EB=A5=98=EB=A1=9C=20=EC=9D=B8=ED=95=9C=20?= =?UTF-8?q?=EC=BD=94=EB=93=9C=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../com/dku/council/domain/chat/service/MessageReceiver.java | 5 ++++- .../com/dku/council/domain/chat/service/MessageSender.java | 5 ++++- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/src/main/java/com/dku/council/domain/chat/service/MessageReceiver.java b/src/main/java/com/dku/council/domain/chat/service/MessageReceiver.java index e5f4a773..59b69870 100644 --- a/src/main/java/com/dku/council/domain/chat/service/MessageReceiver.java +++ b/src/main/java/com/dku/council/domain/chat/service/MessageReceiver.java @@ -1,9 +1,10 @@ package com.dku.council.domain.chat.service; import com.dku.council.domain.chat.model.dto.Message; -import com.dku.council.domain.chat.model.dto.response.ResponseChatDto; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.SerializationFeature; +import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule; import lombok.RequiredArgsConstructor; import lombok.extern.slf4j.Slf4j; import org.springframework.kafka.annotation.KafkaListener; @@ -20,6 +21,8 @@ public class MessageReceiver { public void receiveMessage(String stringChat) throws JsonProcessingException { ObjectMapper objectMapper = new ObjectMapper(); + objectMapper.registerModule(new JavaTimeModule()); + objectMapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS); Message message = objectMapper.readValue(stringChat, Message.class); log.info("Consumed Message : " + stringChat); diff --git a/src/main/java/com/dku/council/domain/chat/service/MessageSender.java b/src/main/java/com/dku/council/domain/chat/service/MessageSender.java index 83b432c0..4a8546a9 100644 --- a/src/main/java/com/dku/council/domain/chat/service/MessageSender.java +++ b/src/main/java/com/dku/council/domain/chat/service/MessageSender.java @@ -1,9 +1,10 @@ package com.dku.council.domain.chat.service; import com.dku.council.domain.chat.model.dto.Message; -import com.dku.council.domain.chat.model.dto.response.ResponseChatDto; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.SerializationFeature; +import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule; import lombok.RequiredArgsConstructor; import lombok.extern.slf4j.Slf4j; import org.springframework.kafka.core.KafkaTemplate; @@ -20,6 +21,8 @@ public void send(String topic, Message data) { // 메시지를 KafkaTemplate 를 사용하여 지정된 토픽으로 전송 ObjectMapper objectMapper = new ObjectMapper(); + objectMapper.registerModule(new JavaTimeModule()); + objectMapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS); try { String stringChat = objectMapper.writeValueAsString(data); log.info("MessageSender Message -> String형 : " + stringChat); From cb38063917bd3f2887180af08b79b70bc7849d04 Mon Sep 17 00:00:00 2001 From: Jungwoo Kim Date: Thu, 1 Feb 2024 21:25:48 +0900 Subject: [PATCH 2/2] =?UTF-8?q?feat:=20=EC=B1=84=ED=8C=85=EB=B0=A9=20?= =?UTF-8?q?=EB=A9=94=EC=8B=9C=EC=A7=80=EC=97=90=20=EC=8B=9C=EA=B0=84=20?= =?UTF-8?q?=ED=91=9C=EC=8B=9C=20=EA=B8=B0=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../chat/controller/ChatController.java | 19 +++++++----- .../domain/chat/model/dto/Message.java | 8 ++++- .../service/ChatRoomMessageService.java | 5 +-- .../resources/static/js/chatroom/socket.js | 31 +++++++++++++++++-- 4 files changed, 50 insertions(+), 13 deletions(-) diff --git a/src/main/java/com/dku/council/domain/chat/controller/ChatController.java b/src/main/java/com/dku/council/domain/chat/controller/ChatController.java index b1b0308f..356d9689 100644 --- a/src/main/java/com/dku/council/domain/chat/controller/ChatController.java +++ b/src/main/java/com/dku/council/domain/chat/controller/ChatController.java @@ -27,6 +27,7 @@ import org.springframework.web.bind.annotation.ResponseBody; import org.springframework.web.socket.messaging.SessionDisconnectEvent; +import java.time.LocalDateTime; import java.util.List; @Tag(name = "채팅", description = "채팅 송/수신 관련 api") @@ -66,9 +67,6 @@ public void enterUser(@Payload RequestChatDto chat, // 채팅방에 유저 추가 및 UserUUID 반환 String username = chatService.addUser(chat.getRoomId(), chat.getSender()); - log.info("enterUser에서 uuid " + username); - log.info("enterUser에서 roomId " + chat.getRoomId()); - log.info("enterUser에서 userId " + chat.getUserId()); // 반환 결과를 socket session 에 userUUID 로 저장 headerAccessor.getSessionAttributes().put("username", username); @@ -76,14 +74,17 @@ public void enterUser(@Payload RequestChatDto chat, headerAccessor.getSessionAttributes().put("roomId", chat.getRoomId()); String enterMessage = chat.getSender() + " 님 입장!!"; + LocalDateTime messageTime = LocalDateTime.now(); + // 입장 메시지 저장 - chatRoomMessageService.create(chat.getRoomId(), chat.getType().toString(), chat.getUserId(), chat.getSender(), enterMessage); + chatRoomMessageService.create(chat.getRoomId(), chat.getType().toString(), chat.getUserId(), chat.getSender(), enterMessage, messageTime); Message message = Message.builder() .type(chat.getType()) .roomId(chat.getRoomId()) .sender(chat.getSender()) .message(enterMessage) + .messageTime(messageTime) .build(); sender.send(topic, message); @@ -96,15 +97,16 @@ public void enterUser(@Payload RequestChatDto chat, */ @MessageMapping("/chat/sendMessage") public void sendMessage(@Payload RequestChatDto chat) { - log.info("CHAT {}", chat); + LocalDateTime messageTime = LocalDateTime.now(); - chatRoomMessageService.create(chat.getRoomId(), chat.getType().toString(), chat.getUserId(), chat.getSender(), chat.getMessage()); + chatRoomMessageService.create(chat.getRoomId(), chat.getType().toString(), chat.getUserId(), chat.getSender(), chat.getMessage(), messageTime); Message message = Message.builder() .type(chat.getType()) .roomId(chat.getRoomId()) .sender(chat.getSender()) .message(chat.getMessage()) + .messageTime(messageTime) .build(); sender.send(topic, message); @@ -140,14 +142,17 @@ public void webSocketDisconnectListener(SessionDisconnectEvent event) { log.info("User Disconnected : ", username); String exitMessage = username + " 님 퇴장!!"; + LocalDateTime messageTime = LocalDateTime.now(); + // 퇴장 메시지 저장 - chatRoomMessageService.create(roomId, MessageType.LEAVE.toString(), userId, username, exitMessage); + chatRoomMessageService.create(roomId, MessageType.LEAVE.toString(), userId, username, exitMessage, messageTime); // builder 어노테이션 활용 Message message = Message.builder() .type(MessageType.LEAVE) .sender(username) .roomId(roomId) .message(exitMessage) + .messageTime(messageTime) .build(); sender.send(topic, message); diff --git a/src/main/java/com/dku/council/domain/chat/model/dto/Message.java b/src/main/java/com/dku/council/domain/chat/model/dto/Message.java index 545848db..dba682af 100644 --- a/src/main/java/com/dku/council/domain/chat/model/dto/Message.java +++ b/src/main/java/com/dku/council/domain/chat/model/dto/Message.java @@ -7,6 +7,7 @@ import lombok.ToString; import javax.validation.constraints.NotNull; +import java.time.LocalDateTime; @Getter @NoArgsConstructor @@ -24,14 +25,19 @@ public class Message { @NotNull private String message; + @NotNull + private LocalDateTime messageTime; + @Builder private Message(MessageType type, String roomId, String sender, - String message) { + String message, + LocalDateTime messageTime) { this.type = type; this.roomId = roomId; this.sender = sender; this.message = message; + this.messageTime = messageTime; } } diff --git a/src/main/java/com/dku/council/domain/chatmessage/service/ChatRoomMessageService.java b/src/main/java/com/dku/council/domain/chatmessage/service/ChatRoomMessageService.java index ceceb31f..eb553566 100644 --- a/src/main/java/com/dku/council/domain/chatmessage/service/ChatRoomMessageService.java +++ b/src/main/java/com/dku/council/domain/chatmessage/service/ChatRoomMessageService.java @@ -23,7 +23,8 @@ public void create(String roomId, String messageType, Long userId, String userNickname, - String content) { + String content, + LocalDateTime messageTime) { ChatRoomMessage chatRoomMessage = new ChatRoomMessage(); chatRoomMessage.setRoomId(roomId); @@ -31,7 +32,7 @@ public void create(String roomId, chatRoomMessage.setUserId(userId); chatRoomMessage.setUserNickname(userNickname); chatRoomMessage.setContent(content); - chatRoomMessage.setCreatedAt(LocalDateTime.now().atZone(seoulZoneId).toLocalDateTime()); + chatRoomMessage.setCreatedAt(messageTime.atZone(seoulZoneId).toLocalDateTime()); chatRoomMessageRepository.save(chatRoomMessage); } diff --git a/src/main/resources/static/js/chatroom/socket.js b/src/main/resources/static/js/chatroom/socket.js index 2c3cd27c..75d764c9 100644 --- a/src/main/resources/static/js/chatroom/socket.js +++ b/src/main/resources/static/js/chatroom/socket.js @@ -107,7 +107,7 @@ function getPreviousMessageList() { sender: data[i]["userNickname"], message: data[i]["content"], type: data[i]["messageType"], - createdAt: data[i]["createdAt"] + messageTime: data[i]["createdAt"] }; console.log(previousChatMessage); previousMessageReceived(JSON.stringify(previousChatMessage)); @@ -159,18 +159,24 @@ function previousMessageReceived(message) { } function addMessageToTheChatRoom(chat, messageElement) { - if (chat.type === 'ENTER' || chat.type === 'LEAVE') { // chatType 이 enter 라면 아래 내용 + if (chat.type === 'ENTER' || chat.type === 'LEAVE') { // chatType 이 enter, leave 라면 아래 내용 messageElement.classList.add('event-message'); getUserList(); } else { // chatType 이 talk 라면 아래 내용 messageElement.classList.add('chat-message'); + let timeElement = document.createElement('span'); + let timeText = document.createTextNode(formatLocalDateTime(chat.messageTime)); + timeElement.appendChild(timeText); + messageElement.appendChild(timeElement); + + messageElement.appendChild(document.createElement('br')); + let avatarElement = document.createElement('i'); let avatarText = document.createTextNode(chat.sender[0]); avatarElement.appendChild(avatarText); avatarElement.style['background-color'] = getAvatarColor(chat.sender); - messageElement.appendChild(avatarElement); let usernameElement = document.createElement('span'); @@ -223,6 +229,25 @@ function getAvatarColor(messageSender) { return colors[index]; } +function formatLocalDateTime(dateTimeString) { + // 주어진 문자열을 Date 객체로 변환 + const dateTime = new Date(dateTimeString); + + // 시간대 추출 + const hours = dateTime.getHours(); + const minutes = dateTime.getMinutes(); + + // 오전/오후 구분 + const meridiem = hours >= 12 ? "오후" : "오전"; + + // 시간 변환 (24시간 형식에서 12시간 형식으로 변환) + const formattedHours = hours % 12 || 12; + + // 시간대와 시간을 문자열로 결합 + return `${meridiem} ${formattedHours.toString().padStart(2, '0')}:${minutes.toString().padStart(2, '0')}`; +} + + document.addEventListener('DOMContentLoaded', function () { connect(); });