diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/app/di/DataSourceModule.kt b/app/src/main/java/com/hackathon/alddeul_babsang/app/di/DataSourceModule.kt index e91e6f3..99fa87a 100644 --- a/app/src/main/java/com/hackathon/alddeul_babsang/app/di/DataSourceModule.kt +++ b/app/src/main/java/com/hackathon/alddeul_babsang/app/di/DataSourceModule.kt @@ -1,10 +1,12 @@ package com.hackathon.alddeul_babsang.app.di import com.hackathon.alddeul_babsang.data.datasource.BabsangDataSource +import com.hackathon.alddeul_babsang.data.datasource.DetailDataSource import com.hackathon.alddeul_babsang.data.datasource.ExampleDataSource import com.hackathon.alddeul_babsang.data.datasource.ProfileDataSource import com.hackathon.alddeul_babsang.data.datasource.UserPreferencesDataSource import com.hackathon.alddeul_babsang.data.datasourceimpl.BabsangDataSourceImpl +import com.hackathon.alddeul_babsang.data.datasourceimpl.DetailDataSourceImpl import com.hackathon.alddeul_babsang.data.datasourceimpl.ExampleDataSourceImpl import com.hackathon.alddeul_babsang.data.datasourceimpl.ProfileDataSourceImpl import com.hackathon.alddeul_babsang.data.datasourceimpl.UserPreferencesDataSourceImpl @@ -33,4 +35,8 @@ abstract class DataSourceModule { @Binds @Singleton abstract fun bindProfileDataSource(profileDataSourceImpl: ProfileDataSourceImpl): ProfileDataSource + + @Binds + @Singleton + abstract fun bindDetailDataSource(detailDataSourceImpl: DetailDataSourceImpl): DetailDataSource } \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/app/di/RepositoryModule.kt b/app/src/main/java/com/hackathon/alddeul_babsang/app/di/RepositoryModule.kt index 18ad144..8fd7c27 100644 --- a/app/src/main/java/com/hackathon/alddeul_babsang/app/di/RepositoryModule.kt +++ b/app/src/main/java/com/hackathon/alddeul_babsang/app/di/RepositoryModule.kt @@ -1,10 +1,12 @@ package com.hackathon.alddeul_babsang.app.di import com.hackathon.alddeul_babsang.data.repositoryimpl.BabsangRepositoryImpl +import com.hackathon.alddeul_babsang.data.repositoryimpl.DetailRepositoryImpl import com.hackathon.alddeul_babsang.data.repositoryimpl.ExampleRepositoryImpl import com.hackathon.alddeul_babsang.data.repositoryimpl.ProfileRepositoryImpl import com.hackathon.alddeul_babsang.data.repositoryimpl.UserPreferencesRepositoryImpl import com.hackathon.alddeul_babsang.domain.repository.BabsangRepository +import com.hackathon.alddeul_babsang.domain.repository.DetailRepository import com.hackathon.alddeul_babsang.domain.repository.ExampleRepository import com.hackathon.alddeul_babsang.domain.repository.ProfileRepository import com.hackathon.alddeul_babsang.domain.repository.UserPreferencesRepository @@ -33,4 +35,8 @@ abstract class RepositoryModule { @Binds @Singleton abstract fun bindProfileRepository(profileRepositoryImpl: ProfileRepositoryImpl): ProfileRepository + + @Binds + @Singleton + abstract fun bindDetailRepository(detailRepositoryImpl: DetailRepositoryImpl): DetailRepository } \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/app/di/ServiceModule.kt b/app/src/main/java/com/hackathon/alddeul_babsang/app/di/ServiceModule.kt index 4509be1..e10276a 100644 --- a/app/src/main/java/com/hackathon/alddeul_babsang/app/di/ServiceModule.kt +++ b/app/src/main/java/com/hackathon/alddeul_babsang/app/di/ServiceModule.kt @@ -1,6 +1,7 @@ package com.hackathon.alddeul_babsang.app.di import com.hackathon.alddeul_babsang.data.service.BabsangApiService +import com.hackathon.alddeul_babsang.data.service.DetailApiService import com.hackathon.alddeul_babsang.data.service.ProfileApiService import com.sopt.data.service.ExampleApiService import dagger.Module @@ -31,4 +32,10 @@ object ServiceModule { fun provideProfileService( @AlddeulRetrofit retrofit: Retrofit ): ProfileApiService = retrofit.create(ProfileApiService::class.java) + + @Provides + @Singleton + fun provideDetailService( + @AlddeulRetrofit retrofit: Retrofit + ): DetailApiService = retrofit.create(DetailApiService::class.java) } diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/data/datasource/DetailDataSource.kt b/app/src/main/java/com/hackathon/alddeul_babsang/data/datasource/DetailDataSource.kt new file mode 100644 index 0000000..38646b0 --- /dev/null +++ b/app/src/main/java/com/hackathon/alddeul_babsang/data/datasource/DetailDataSource.kt @@ -0,0 +1,14 @@ +package com.hackathon.alddeul_babsang.data.datasource + +import com.hackathon.alddeul_babsang.data.dto.BaseResponse +import com.hackathon.alddeul_babsang.data.dto.response.ResponseReviewDto +import okhttp3.MultipartBody +import okhttp3.RequestBody + +interface DetailDataSource { + suspend fun postReview( + storeId: Long, + data: Map, // JSON 데이터를 포함하는 Map + reviewImage: MultipartBody.Part? = null // 이미지 파일 + ): BaseResponse +} \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/data/datasourceimpl/DetailDataSourceImpl.kt b/app/src/main/java/com/hackathon/alddeul_babsang/data/datasourceimpl/DetailDataSourceImpl.kt new file mode 100644 index 0000000..74769ba --- /dev/null +++ b/app/src/main/java/com/hackathon/alddeul_babsang/data/datasourceimpl/DetailDataSourceImpl.kt @@ -0,0 +1,22 @@ +package com.hackathon.alddeul_babsang.data.datasourceimpl + +import com.hackathon.alddeul_babsang.data.datasource.DetailDataSource +import com.hackathon.alddeul_babsang.data.dto.BaseResponse +import com.hackathon.alddeul_babsang.data.dto.response.ResponseReviewDto +import com.hackathon.alddeul_babsang.data.service.DetailApiService +import okhttp3.MultipartBody +import okhttp3.RequestBody +import javax.inject.Inject + +class DetailDataSourceImpl @Inject constructor( + private val detailApiService: DetailApiService +) : DetailDataSource { + override suspend fun postReview( + storeId: Long, + data: Map, + reviewImage: MultipartBody.Part? + ): BaseResponse { + return detailApiService.postReview(storeId, data, reviewImage) + } + +} \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/data/dto/response/ResponseLikesDto.kt b/app/src/main/java/com/hackathon/alddeul_babsang/data/dto/response/ResponseLikesDto.kt index 43e0de2..a173239 100644 --- a/app/src/main/java/com/hackathon/alddeul_babsang/data/dto/response/ResponseLikesDto.kt +++ b/app/src/main/java/com/hackathon/alddeul_babsang/data/dto/response/ResponseLikesDto.kt @@ -5,7 +5,7 @@ import kotlinx.serialization.Serializable @Serializable data class ResponseLikesDto( - @SerialName("favoriteRestaurants") val favoriteRestaurants: List + @SerialName("favoriteStoreDetailDtos") val favoriteRestaurants: List ) @Serializable diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/data/dto/response/ResponseReviewDto.kt b/app/src/main/java/com/hackathon/alddeul_babsang/data/dto/response/ResponseReviewDto.kt new file mode 100644 index 0000000..22b73c1 --- /dev/null +++ b/app/src/main/java/com/hackathon/alddeul_babsang/data/dto/response/ResponseReviewDto.kt @@ -0,0 +1,11 @@ +package com.hackathon.alddeul_babsang.data.dto.response + +import kotlinx.serialization.SerialName +import kotlinx.serialization.Serializable + +@Serializable +data class ResponseReviewDto ( + @SerialName("storeId") val storeId: Long, + @SerialName("userId") val userId: Long, + @SerialName("message") val message: String, +) \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/data/repositoryimpl/DetailRepositoryImpl.kt b/app/src/main/java/com/hackathon/alddeul_babsang/data/repositoryimpl/DetailRepositoryImpl.kt new file mode 100644 index 0000000..2ac1915 --- /dev/null +++ b/app/src/main/java/com/hackathon/alddeul_babsang/data/repositoryimpl/DetailRepositoryImpl.kt @@ -0,0 +1,43 @@ +package com.hackathon.alddeul_babsang.data.repositoryimpl + +import com.hackathon.alddeul_babsang.data.datasource.DetailDataSource +import com.hackathon.alddeul_babsang.domain.repository.DetailRepository +import okhttp3.MediaType.Companion.toMediaTypeOrNull +import okhttp3.MultipartBody +import okhttp3.RequestBody.Companion.asRequestBody +import okhttp3.RequestBody.Companion.toRequestBody +import org.json.JSONObject +import java.io.File +import javax.inject.Inject + +class DetailRepositoryImpl @Inject constructor( + private val detailDataSource: DetailDataSource +) : DetailRepository { + override suspend fun postReview( + storeId: Long, + userId: Long, + rating: Double, + content: String, + reviewImage: File + ): Result { + return runCatching { + val dataMap = mapOf( + "userId" to userId.toString().toRequestBody("text/plain".toMediaTypeOrNull()), + "rating" to rating.toString().toRequestBody("text/plain".toMediaTypeOrNull()), + "content" to content.toRequestBody("text/plain".toMediaTypeOrNull()) + ) + + // 이미지 파일 파트 생성 + val filePart = reviewImage?.let { + val requestBody = it.asRequestBody("image/jpeg".toMediaTypeOrNull()) + MultipartBody.Part.createFormData("reviewImage", it.name, requestBody) + } + + detailDataSource.postReview( + storeId = storeId, + data = dataMap, + reviewImage = filePart + ).message + } + } +} \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/data/service/ApiKeyStorage.kt b/app/src/main/java/com/hackathon/alddeul_babsang/data/service/ApiKeyStorage.kt index cf391b3..20eec7a 100644 --- a/app/src/main/java/com/hackathon/alddeul_babsang/data/service/ApiKeyStorage.kt +++ b/app/src/main/java/com/hackathon/alddeul_babsang/data/service/ApiKeyStorage.kt @@ -19,4 +19,5 @@ object ApiKeyStorage { const val MYPAGE = "mypage" const val LOGOUT = "logout" const val USER_ID = "userId" + const val STORE_ID = "storeId" } \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/data/service/DetailApiService.kt b/app/src/main/java/com/hackathon/alddeul_babsang/data/service/DetailApiService.kt new file mode 100644 index 0000000..395383d --- /dev/null +++ b/app/src/main/java/com/hackathon/alddeul_babsang/data/service/DetailApiService.kt @@ -0,0 +1,23 @@ +package com.hackathon.alddeul_babsang.data.service + +import com.hackathon.alddeul_babsang.data.dto.BaseResponse +import com.hackathon.alddeul_babsang.data.dto.response.ResponseReviewDto +import com.sopt.data.service.ApiKeyStorage.REVIEW +import com.sopt.data.service.ApiKeyStorage.STORE_ID +import okhttp3.MultipartBody +import okhttp3.RequestBody +import retrofit2.http.Multipart +import retrofit2.http.POST +import retrofit2.http.Part +import retrofit2.http.PartMap +import retrofit2.http.Path + +interface DetailApiService { + @Multipart + @POST("/$REVIEW/{$STORE_ID}") + suspend fun postReview( + @Path("storeId") storeId: Long, + @PartMap data: Map, + @Part reviewImage: MultipartBody.Part? = null + ): BaseResponse +} \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/data/service/ProfileApiService.kt b/app/src/main/java/com/hackathon/alddeul_babsang/data/service/ProfileApiService.kt index 9a3b966..886aeb3 100644 --- a/app/src/main/java/com/hackathon/alddeul_babsang/data/service/ProfileApiService.kt +++ b/app/src/main/java/com/hackathon/alddeul_babsang/data/service/ProfileApiService.kt @@ -5,6 +5,7 @@ import com.hackathon.alddeul_babsang.data.dto.request.RequestLikesDto import com.hackathon.alddeul_babsang.data.dto.response.ResponseLikesDto import com.sopt.data.service.ApiKeyStorage.FAVORITES import com.sopt.data.service.ApiKeyStorage.USER_ID +import retrofit2.http.Body import retrofit2.http.GET import retrofit2.http.POST import retrofit2.http.Path @@ -17,6 +18,6 @@ interface ProfileApiService { @POST("/$FAVORITES") suspend fun postLike( - requestLikesDto: RequestLikesDto + @Body requestLikesDto: RequestLikesDto ): BaseResponse } \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/domain/repository/DetailRepository.kt b/app/src/main/java/com/hackathon/alddeul_babsang/domain/repository/DetailRepository.kt new file mode 100644 index 0000000..ffd6040 --- /dev/null +++ b/app/src/main/java/com/hackathon/alddeul_babsang/domain/repository/DetailRepository.kt @@ -0,0 +1,13 @@ +package com.hackathon.alddeul_babsang.domain.repository + +import java.io.File + +interface DetailRepository { + suspend fun postReview( + storeId: Long, + userId: Long, + rating: Double, + content: String, + reviewImage: File + ): Result +} \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/presentation/babsang/screen/BabsangItem.kt b/app/src/main/java/com/hackathon/alddeul_babsang/presentation/babsang/screen/BabsangItem.kt index fe9e5ff..bac5a52 100644 --- a/app/src/main/java/com/hackathon/alddeul_babsang/presentation/babsang/screen/BabsangItem.kt +++ b/app/src/main/java/com/hackathon/alddeul_babsang/presentation/babsang/screen/BabsangItem.kt @@ -159,22 +159,4 @@ fun ReplaceImage2(codeName: String, imageUrl: String?) { ) } } -} - -@Preview(showBackground = true) -@Composable -fun BabsangListItemPreview() { - AlddeulBabsangTheme { - LikeItem( - data = LikesEntity( - id = 1, - avatar = null, - name = "송이네 밥상", - codeName = "경양식/일식", - address = "서울특별시 용산구 청파동 11", - phone = "02-210-0220", - favorite = true - ) - ) - } } \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/presentation/detail/screen/ReviewScreen.kt b/app/src/main/java/com/hackathon/alddeul_babsang/presentation/detail/screen/ReviewScreen.kt index fb8b0bb..6446db3 100644 --- a/app/src/main/java/com/hackathon/alddeul_babsang/presentation/detail/screen/ReviewScreen.kt +++ b/app/src/main/java/com/hackathon/alddeul_babsang/presentation/detail/screen/ReviewScreen.kt @@ -25,6 +25,7 @@ import androidx.compose.material3.TopAppBarDefaults import androidx.compose.runtime.Composable import androidx.compose.runtime.SideEffect import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableIntStateOf import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember import androidx.compose.runtime.setValue @@ -39,6 +40,8 @@ import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.vectorResource import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.dp +import androidx.hilt.navigation.compose.hiltViewModel +import androidx.lifecycle.compose.collectAsStateWithLifecycle import coil.compose.rememberAsyncImagePainter import com.google.accompanist.systemuicontroller.rememberSystemUiController import com.hackathon.alddeul_babsang.R @@ -55,15 +58,36 @@ import com.hackathon.alddeul_babsang.core_ui.theme.Yellow import com.hackathon.alddeul_babsang.core_ui.theme.body4Regular import com.hackathon.alddeul_babsang.core_ui.theme.head4Bold import com.hackathon.alddeul_babsang.presentation.detail.navigation.DetailNavigator +import com.hackathon.alddeul_babsang.util.UiState import com.hackathon.alddeul_babsang.util.toast +import com.hackathon.alddeul_babsang.util.uriToFile +import timber.log.Timber @Composable fun ReviewRoute( navigator: DetailNavigator, id: Long, ) { + val reviewViewModel: ReviewViewModel = hiltViewModel() val keyboardController = LocalSoftwareKeyboardController.current val systemUiController = rememberSystemUiController() + val postReviewState by reviewViewModel.postReviewState.collectAsStateWithLifecycle(UiState.Empty) + + when (postReviewState) { + is UiState.Success -> { + navigator.navigateBack() + keyboardController?.hide() + Timber.d("Review post success") + } + + is UiState.Failure -> { + val message = (postReviewState as UiState.Failure).msg + Timber.e("Review post failed: $message") + LocalContext.current.toast(message) + } + + else -> {} + } SideEffect { systemUiController.setStatusBarColor( @@ -72,19 +96,18 @@ fun ReviewRoute( } ReviewScreen( + id = id, onBackClick = { navigator.navigateBack() }, - onCompleteClick = { - keyboardController?.hide() - navigator.navigateBack() - } + reviewViewModel = reviewViewModel ) } @OptIn(ExperimentalMaterial3Api::class) @Composable fun ReviewScreen( + id: Long = 0, onBackClick: () -> Unit = {}, - onCompleteClick: () -> Unit = {} + reviewViewModel: ReviewViewModel ) { var imageUri by remember { mutableStateOf(null) } val galleryLauncher = rememberLauncherForActivityResult( @@ -95,6 +118,7 @@ fun ReviewScreen( ) var reviewLength by remember { mutableStateOf("") } val context = LocalContext.current + var rating by remember { mutableIntStateOf(0) } Scaffold( topBar = { @@ -158,7 +182,10 @@ fun ReviewScreen( } Spacer(modifier = Modifier.height(30.dp)) AlddeulHeader(text = R.string.tv_review_rating) - StarRating() + StarRating( + rating = rating, + onRatingChange = { rating = it } + ) AlddeulHeader(text = R.string.tv_review_detail) ReviewTextField( value = reviewLength, @@ -177,8 +204,16 @@ fun ReviewScreen( AlddeulButton( text = R.string.btn_review_complete, onClick = { - if (reviewLength.length <= 100) onCompleteClick() - else context.toast(context.getString(R.string.toast_review_length)) + if (reviewLength.length <= 100 && imageUri != null) { + val file = uriToFile(imageUri!!, context) + reviewViewModel.postReview( + storeId = id, + userId = 1, + rating = rating.toDouble(), + content = reviewLength, + reviewImage = file + ) + } else context.toast(context.getString(R.string.toast_review_length)) } ) } @@ -186,21 +221,23 @@ fun ReviewScreen( } @Composable -fun StarRating() { +fun StarRating( + rating: Int, // 현재 선택된 별 개수 + onRatingChange: (Int) -> Unit // 별 개수 변경 시 호출되는 함수 +) { val starCount = 5 - var selectedStars by remember { mutableStateOf(List(starCount) { false }) } LazyRow { items(starCount) { index -> IconButton( onClick = { - selectedStars = List(selectedStars.size) { i -> i <= index } + onRatingChange(index + 1) // 클릭된 별까지의 개수를 전달 } ) { Icon( imageVector = ImageVector.vectorResource(id = R.drawable.ic_review_star), contentDescription = null, - tint = if (selectedStars[index]) Yellow else Gray100 + tint = if (index < rating) Yellow else Gray100 ) } } @@ -211,6 +248,8 @@ fun StarRating() { @Composable fun ReviewScreenPreview() { AlddeulBabsangTheme { - ReviewScreen() + ReviewScreen( + reviewViewModel = hiltViewModel() + ) } } \ No newline at end of file diff --git a/app/src/main/java/com/hackathon/alddeul_babsang/presentation/detail/screen/ReviewViewModel.kt b/app/src/main/java/com/hackathon/alddeul_babsang/presentation/detail/screen/ReviewViewModel.kt new file mode 100644 index 0000000..309897c --- /dev/null +++ b/app/src/main/java/com/hackathon/alddeul_babsang/presentation/detail/screen/ReviewViewModel.kt @@ -0,0 +1,38 @@ +package com.hackathon.alddeul_babsang.presentation.detail.screen + +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import com.hackathon.alddeul_babsang.domain.repository.DetailRepository +import com.hackathon.alddeul_babsang.util.UiState +import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.launch +import java.io.File +import javax.inject.Inject + +@HiltViewModel +class ReviewViewModel @Inject constructor( + private val detailRepository: DetailRepository +) : ViewModel() { + private val _postReviewState = MutableStateFlow>(UiState.Empty) + val postReviewState: StateFlow> = _postReviewState + + fun postReview( + storeId: Long, + userId: Long, + rating: Double, + content: String, + reviewImage: File + ) = viewModelScope.launch { + _postReviewState.emit(UiState.Loading) + detailRepository.postReview(storeId, userId, rating, content, reviewImage).fold( + onSuccess = { + _postReviewState.emit(UiState.Success(it)) + }, + onFailure = { + _postReviewState.emit(UiState.Failure(it.message.toString())) + } + ) + } +} \ No newline at end of file