diff --git a/android/festago/data/build.gradle.kts b/android/festago/data/build.gradle.kts index 03e2336ec..912a80fbb 100644 --- a/android/festago/data/build.gradle.kts +++ b/android/festago/data/build.gradle.kts @@ -78,6 +78,13 @@ dependencies { implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.4") implementation("org.jetbrains.kotlinx:kotlinx-coroutines-jdk8:1.6.4") testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-test:1.6.4") + + // room + val room_version = "2.6.1" + implementation("androidx.room:room-runtime:$room_version") + annotationProcessor("androidx.room:room-compiler:$room_version") + kapt("androidx.room:room-compiler:$room_version") + implementation("androidx.room:room-ktx:$room_version") } fun getSecretKey(propertyKey: String): String { diff --git a/android/festago/data/src/main/java/com/festago/festago/data/dao/RecentSearchQueryDao.kt b/android/festago/data/src/main/java/com/festago/festago/data/dao/RecentSearchQueryDao.kt new file mode 100644 index 000000000..611889f8c --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/dao/RecentSearchQueryDao.kt @@ -0,0 +1,22 @@ +package com.festago.festago.data.dao + +import androidx.room.Dao +import androidx.room.Delete +import androidx.room.Query +import androidx.room.Upsert +import com.festago.festago.data.model.RecentSearchQueryEntity + +@Dao +interface RecentSearchQueryDao { + @Query(value = "SELECT * FROM recentSearchQueries ORDER BY created_at DESC LIMIT :limit") + suspend fun getRecentSearchQueryEntities(limit: Int): List + + @Upsert + suspend fun insertOrReplaceRecentSearchQuery(recentSearchQuery: RecentSearchQueryEntity) + + @Delete + suspend fun deleteRecentSearchQuery(recentSearchQuery: RecentSearchQueryEntity) + + @Query(value = "DELETE FROM recentSearchQueries") + suspend fun clearRecentSearchQueries() +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/database/FestagoDatabase.kt b/android/festago/data/src/main/java/com/festago/festago/data/database/FestagoDatabase.kt new file mode 100644 index 000000000..6db6158bf --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/database/FestagoDatabase.kt @@ -0,0 +1,11 @@ +package com.festago.festago.data.database + +import androidx.room.Database +import androidx.room.RoomDatabase +import com.festago.festago.data.dao.RecentSearchQueryDao +import com.festago.festago.data.model.RecentSearchQueryEntity + +@Database(entities = [RecentSearchQueryEntity::class], version = 1) +abstract class FestagoDatabase : RoomDatabase() { + abstract fun recentSearchQueryDao(): RecentSearchQueryDao +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/DaosModule.kt b/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/DaosModule.kt new file mode 100644 index 000000000..6ad5ca163 --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/DaosModule.kt @@ -0,0 +1,17 @@ +package com.festago.festago.data.di.singletonscope + +import com.festago.festago.data.dao.RecentSearchQueryDao +import com.festago.festago.data.database.FestagoDatabase +import dagger.Module +import dagger.Provides +import dagger.hilt.InstallIn +import dagger.hilt.components.SingletonComponent + +@Module +@InstallIn(SingletonComponent::class) +object DaosModule { + + @Provides + fun providesRecentSearchQueryDao(database: FestagoDatabase): RecentSearchQueryDao = + database.recentSearchQueryDao() +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/DatabaseModule.kt b/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/DatabaseModule.kt new file mode 100644 index 000000000..2c4e9c218 --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/DatabaseModule.kt @@ -0,0 +1,25 @@ +package com.festago.festago.data.di.singletonscope + +import android.content.Context +import androidx.room.Room +import com.festago.festago.data.database.FestagoDatabase +import dagger.Module +import dagger.Provides +import dagger.hilt.InstallIn +import dagger.hilt.android.qualifiers.ApplicationContext +import dagger.hilt.components.SingletonComponent +import javax.inject.Singleton + +@Module +@InstallIn(SingletonComponent::class) +object DatabaseModule { + @Provides + @Singleton + fun providesFestagoDatabase( + @ApplicationContext context: Context, + ): FestagoDatabase = Room.databaseBuilder( + context, + FestagoDatabase::class.java, + "festago-database", + ).build() +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/RepositoryModule.kt b/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/RepositoryModule.kt index 2ca7f616d..556bcf3da 100644 --- a/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/RepositoryModule.kt +++ b/android/festago/data/src/main/java/com/festago/festago/data/di/singletonscope/RepositoryModule.kt @@ -1,11 +1,15 @@ package com.festago.festago.data.di.singletonscope +import com.festago.festago.data.repository.DefaultRecentSearchRepository import com.festago.festago.data.repository.FakeArtistRepository import com.festago.festago.data.repository.FakeFestivalRepository import com.festago.festago.data.repository.FakeSchoolRepository +import com.festago.festago.data.repository.FakeSearchRepository import com.festago.festago.domain.repository.ArtistRepository import com.festago.festago.domain.repository.FestivalRepository +import com.festago.festago.domain.repository.RecentSearchRepository import com.festago.festago.domain.repository.SchoolRepository +import com.festago.festago.domain.repository.SearchRepository import dagger.Binds import dagger.Module import dagger.hilt.InstallIn @@ -27,4 +31,12 @@ interface RepositoryModule { @Binds @Singleton fun bindsSchoolRepository(schoolRepository: FakeSchoolRepository): SchoolRepository + + @Binds + @Singleton + fun bindsRecentSearchRepository(recentSearchRepository: DefaultRecentSearchRepository): RecentSearchRepository + + @Binds + @Singleton + fun bindsSearchRepository(searchRepository: FakeSearchRepository): SearchRepository } diff --git a/android/festago/data/src/main/java/com/festago/festago/data/dto/artist/ArtistSearchResponse.kt b/android/festago/data/src/main/java/com/festago/festago/data/dto/artist/ArtistSearchResponse.kt new file mode 100644 index 000000000..098e28470 --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/dto/artist/ArtistSearchResponse.kt @@ -0,0 +1,21 @@ +package com.festago.festago.data.dto.artist + +import com.festago.festago.domain.model.search.ArtistSearch +import kotlinx.serialization.Serializable + +@Serializable +data class ArtistSearchResponse( + val id: Long, + val name: String, + val profileImageUrl: String, + val todayStage: Int, + val upcomingStage: Int, +) { + fun toDomain() = ArtistSearch( + id = id, + name = name, + profileImageUrl = profileImageUrl, + todayStage = todayStage, + upcomingStage = upcomingStage, + ) +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/dto/school/SchoolSearchResponse.kt b/android/festago/data/src/main/java/com/festago/festago/data/dto/school/SchoolSearchResponse.kt new file mode 100644 index 000000000..a8ab759a7 --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/dto/school/SchoolSearchResponse.kt @@ -0,0 +1,20 @@ +package com.festago.festago.data.dto.school + +import com.festago.festago.domain.model.search.SchoolSearch +import kotlinx.serialization.Serializable +import java.time.LocalDate + +@Serializable +data class SchoolSearchResponse( + val id: Long, + val name: String, + val logoUrl: String, + val upcomingFestivalStartDate: String?, +) { + fun toDomain() = SchoolSearch( + id = id, + name = name, + logoUrl = logoUrl, + upcomingFestivalStartDate = upcomingFestivalStartDate?.let { LocalDate.parse(it) }, + ) +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/model/RecentSearchQueryEntity.kt b/android/festago/data/src/main/java/com/festago/festago/data/model/RecentSearchQueryEntity.kt new file mode 100644 index 000000000..2076f4bbe --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/model/RecentSearchQueryEntity.kt @@ -0,0 +1,18 @@ +package com.festago.festago.data.model + +import androidx.room.ColumnInfo +import androidx.room.Entity +import androidx.room.PrimaryKey +import com.festago.festago.domain.model.recentsearch.RecentSearchQuery + +@Entity( + tableName = "recentSearchQueries", +) +data class RecentSearchQueryEntity( + @PrimaryKey + val query: String, + @ColumnInfo(name = "created_at") + val createdAt: Long, +) { + fun toDomain() = RecentSearchQuery(query = query, queriedDate = createdAt) +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/repository/DefaultRecentSearchRepository.kt b/android/festago/data/src/main/java/com/festago/festago/data/repository/DefaultRecentSearchRepository.kt new file mode 100644 index 000000000..f6506e01c --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/repository/DefaultRecentSearchRepository.kt @@ -0,0 +1,39 @@ +package com.festago.festago.data.repository + +import com.festago.festago.data.dao.RecentSearchQueryDao +import com.festago.festago.data.model.RecentSearchQueryEntity +import com.festago.festago.domain.model.recentsearch.RecentSearchQuery +import com.festago.festago.domain.repository.RecentSearchRepository +import kotlinx.coroutines.flow.map +import javax.inject.Inject + +class DefaultRecentSearchRepository @Inject constructor( + private val recentSearchQueryDao: RecentSearchQueryDao, +) : RecentSearchRepository { + + override suspend fun insertOrReplaceRecentSearch(searchQuery: String) { + recentSearchQueryDao.insertOrReplaceRecentSearchQuery( + RecentSearchQueryEntity( + query = searchQuery, + createdAt = System.currentTimeMillis(), + ), + ) + } + + override suspend fun deleteRecentSearch(searchQuery: String) { + recentSearchQueryDao.deleteRecentSearchQuery( + RecentSearchQueryEntity( + query = searchQuery, + createdAt = System.currentTimeMillis(), + ), + ) + } + + override suspend fun getRecentSearchQueries(limit: Int): List { + return recentSearchQueryDao.getRecentSearchQueryEntities(limit).map { recentSearchQueries -> + recentSearchQueries.toDomain() + } + } + + override suspend fun clearRecentSearches() = recentSearchQueryDao.clearRecentSearchQueries() +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/repository/DefaultSearchRepository.kt b/android/festago/data/src/main/java/com/festago/festago/data/repository/DefaultSearchRepository.kt new file mode 100644 index 000000000..322fbcb23 --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/repository/DefaultSearchRepository.kt @@ -0,0 +1,33 @@ +package com.festago.festago.data.repository + +import com.festago.festago.data.service.SearchRetrofitService +import com.festago.festago.data.util.onSuccessOrCatch +import com.festago.festago.data.util.runCatchingResponse +import com.festago.festago.domain.model.festival.Festival +import com.festago.festago.domain.model.search.ArtistSearch +import com.festago.festago.domain.model.search.SchoolSearch +import com.festago.festago.domain.repository.SearchRepository +import javax.inject.Inject + +class DefaultSearchRepository @Inject constructor( + private val searchRetrofitService: SearchRetrofitService, +) : SearchRepository { + + override suspend fun searchFestivals(searchQuery: String): Result> { + return runCatchingResponse { searchRetrofitService.searchFestivals(searchQuery) }.onSuccessOrCatch { festivalResponses -> + festivalResponses.map { it.toDomain() } + } + } + + override suspend fun searchArtists(searchQuery: String): Result> { + return runCatchingResponse { + searchRetrofitService.searchArtists(searchQuery) + }.onSuccessOrCatch { artistSearchResponses -> artistSearchResponses.map { it.toDomain() } } + } + + override suspend fun searchSchools(searchQuery: String): Result> { + return runCatchingResponse { + searchRetrofitService.searchSchools(searchQuery) + }.onSuccessOrCatch { schoolSearchResponses -> schoolSearchResponses.map { it.toDomain() } } + } +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/repository/FakeFestivals.kt b/android/festago/data/src/main/java/com/festago/festago/data/repository/FakeFestivals.kt index 382516c38..fe695f2ee 100644 --- a/android/festago/data/src/main/java/com/festago/festago/data/repository/FakeFestivals.kt +++ b/android/festago/data/src/main/java/com/festago/festago/data/repository/FakeFestivals.kt @@ -391,8 +391,8 @@ object FakeFestivals { Festival( id = 1, name = "뉴진스 콘서트", - startDate = LocalDate.MIN, - endDate = LocalDate.MAX, + startDate = LocalDate.now().minusDays(5L), + endDate = LocalDate.now().minusDays(3L), imageUrl = "", school = School(id = 1L, name = "고려대", imageUrl = ""), artists = listOf( @@ -406,8 +406,8 @@ object FakeFestivals { Festival( id = 2, name = "아이브 콘서트", - startDate = LocalDate.MIN, - endDate = LocalDate.MAX, + startDate = LocalDate.now().minusDays(2L), + endDate = LocalDate.now().plusDays(1L), imageUrl = "", school = School(id = 1L, name = "연세대", imageUrl = ""), artists = listOf( @@ -441,8 +441,8 @@ object FakeFestivals { Festival( id = 3, name = "아이들 콘서트", - startDate = LocalDate.MIN, - endDate = LocalDate.MAX, + startDate = LocalDate.now().plusDays(5L), + endDate = LocalDate.now().plusDays(6L), imageUrl = "", school = School(id = 1L, name = "연세대", imageUrl = ""), artists = listOf( @@ -486,8 +486,8 @@ object FakeFestivals { Festival( id = 5, name = "아이브 콘서트", - startDate = LocalDate.MIN, - endDate = LocalDate.MAX, + startDate = LocalDate.now().plusDays(10L), + endDate = LocalDate.now().plusDays(11L), imageUrl = "", school = School(id = 1L, name = "연세대", imageUrl = ""), artists = listOf( diff --git a/android/festago/data/src/main/java/com/festago/festago/data/repository/FakeSearchRepository.kt b/android/festago/data/src/main/java/com/festago/festago/data/repository/FakeSearchRepository.kt new file mode 100644 index 000000000..bb269124b --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/repository/FakeSearchRepository.kt @@ -0,0 +1,94 @@ +package com.festago.festago.data.repository + +import com.festago.festago.domain.model.festival.Festival +import com.festago.festago.domain.model.search.ArtistSearch +import com.festago.festago.domain.model.search.SchoolSearch +import com.festago.festago.domain.repository.SearchRepository +import kotlinx.coroutines.delay +import java.time.LocalDate +import javax.inject.Inject + +class FakeSearchRepository @Inject constructor() : SearchRepository { + private var times = 0 + override suspend fun searchFestivals(searchQuery: String): Result> { + delay(1000) + times++ + if (times % 2 == 0) { + return Result.success(FakeFestivals.popularFestivals) + } + return Result.success(listOf()) + } + + override suspend fun searchArtists(searchQuery: String): Result> { + delay(1000) + if (times % 5 == 0) { + return Result.failure(Exception()) + } + return Result.success( + listOf( + ArtistSearch( + id = 6L, + name = "뉴진스", + profileImageUrl = "https://cdn.mediatoday.co.kr/news/photo/202311/313885_438531_4716.jpg", + todayStage = 2, + upcomingStage = 1, + ), + ArtistSearch( + id = 1L, + name = "BTS", + profileImageUrl = "https://i.namu.wiki/i/gpgJvt_C2vKJS4VA4K_Vm57Y5WoS83ofshxhJlQaT4P9Tu0N96vZ2OcdeAN7ZtRAM26UyyQs3sualkKk6i_SrRMvwVKrU015XJqzJ7wKRbOub_oUAxPSFre_8D5De3oy-fCxL0uZ-HGvsWxIX57yrw.webp", + todayStage = 2, + upcomingStage = 2, + ), + ArtistSearch( + id = 2L, + name = "싸이", + profileImageUrl = "https://i.namu.wiki/i/VH58lI8f-y8QSoxFH9IAjjCobySN0lflZ4rMy6Un7qawUwAyi9UfeseZWCzxH-lQeZk7q_eUyTHGlZBAPqSLWliIKWYDLaAgomVtOyAQg60aCpF3oNTBOgUe_hig3rbHW-YAgoj95Fww3MCToyM6MA.webp", + todayStage = 2, + upcomingStage = 3, + ), + ArtistSearch( + id = 10L, + name = "마마무", + profileImageUrl = "https://i.namu.wiki/i/Mre8tXnE40mB9_UwXIwASMEAUSVhHvyjJxXq-lQo40C3bLWYfxXBeai8t6TugyomPjFgxL3VfDA2zn65HlzqPXgTKlvdRl1gJ6PGZLxYYk8Uhk8L6va7zm_etSK5UzVLE56fUATqUCq-6tRQXigmYQ.webp", + todayStage = 2, + upcomingStage = 4, + ), + ArtistSearch( + id = 11L, + name = "블랙핑크", + profileImageUrl = "https://i.namu.wiki/i/VZxRYO8_CXa2QbOSZgttDq5ue5QEu_Fbk1Lwo3qpasLAfS802YExcnmVmDhCq3ONF0ExzhACz_YkZbxOGmIfjuPDZnFo7i0pWaT05NluHRHGfp9NqsAT6WBNb0k5KecOyDvakXk0VH2fUo4ojSwC6g.webp", + todayStage = 1, + upcomingStage = 5, + ), + ), + ) + } + + override suspend fun searchSchools(searchQuery: String): Result> { + delay(1000) + return Result.success( + listOf( + SchoolSearch( + id = 1L, + name = "부경대학교", + logoUrl = "htts://www.pknu.ac.kr/images/front/sub/univ_logo00.png", + upcomingFestivalStartDate = LocalDate.now().plusDays(10L), + ), + SchoolSearch( + id = 2L, + name = "서울대학교", + logoUrl = "https://blog.kakaocdn.net/dn/CYoCP/btrSeivmaxD/e7JaOZVPI3Je55nAJaHDMK/img.png", + upcomingFestivalStartDate = LocalDate.now().plusDays(3L), + ), + SchoolSearch( + id = 3L, + name = "서울과학기술대학교", + logoUrl = "https://www.seoultech.ac.kr/site/www/images/intro/img_ui01_01.gif", + upcomingFestivalStartDate = null, + ), + + ), + ) + } +} diff --git a/android/festago/data/src/main/java/com/festago/festago/data/service/SearchRetrofitService.kt b/android/festago/data/src/main/java/com/festago/festago/data/service/SearchRetrofitService.kt new file mode 100644 index 000000000..794b43f9d --- /dev/null +++ b/android/festago/data/src/main/java/com/festago/festago/data/service/SearchRetrofitService.kt @@ -0,0 +1,25 @@ +package com.festago.festago.data.service + +import com.festago.festago.data.dto.artist.ArtistSearchResponse +import com.festago.festago.data.dto.festival.FestivalResponse +import com.festago.festago.data.dto.school.SchoolSearchResponse +import retrofit2.Response +import retrofit2.http.GET +import retrofit2.http.Query + +interface SearchRetrofitService { + @GET("api/v1/search/festivals") + suspend fun searchFestivals( + @Query("keyword") keyword: String, + ): Response> + + @GET("api/v1/search/artists") + suspend fun searchArtists( + @Query("keyword") keyword: String, + ): Response> + + @GET("api/v1/search/schools") + suspend fun searchSchools( + @Query("keyword") keyword: String, + ): Response> +} diff --git a/android/festago/domain/src/main/java/com/festago/festago/domain/model/recentsearch/RecentSearchQuery.kt b/android/festago/domain/src/main/java/com/festago/festago/domain/model/recentsearch/RecentSearchQuery.kt new file mode 100644 index 000000000..197b84bf3 --- /dev/null +++ b/android/festago/domain/src/main/java/com/festago/festago/domain/model/recentsearch/RecentSearchQuery.kt @@ -0,0 +1,6 @@ +package com.festago.festago.domain.model.recentsearch + +data class RecentSearchQuery( + val query: String, + val queriedDate: Long, +) diff --git a/android/festago/domain/src/main/java/com/festago/festago/domain/model/search/ArtistSearch.kt b/android/festago/domain/src/main/java/com/festago/festago/domain/model/search/ArtistSearch.kt new file mode 100644 index 000000000..fb472d087 --- /dev/null +++ b/android/festago/domain/src/main/java/com/festago/festago/domain/model/search/ArtistSearch.kt @@ -0,0 +1,9 @@ +package com.festago.festago.domain.model.search + +data class ArtistSearch( + val id: Long, + val name: String, + val profileImageUrl: String, + val todayStage: Int, + val upcomingStage: Int, +) diff --git a/android/festago/domain/src/main/java/com/festago/festago/domain/model/search/SchoolSearch.kt b/android/festago/domain/src/main/java/com/festago/festago/domain/model/search/SchoolSearch.kt new file mode 100644 index 000000000..dde84b81e --- /dev/null +++ b/android/festago/domain/src/main/java/com/festago/festago/domain/model/search/SchoolSearch.kt @@ -0,0 +1,10 @@ +package com.festago.festago.domain.model.search + +import java.time.LocalDate + +data class SchoolSearch( + val id: Long, + val name: String, + val logoUrl: String, + val upcomingFestivalStartDate: LocalDate?, +) diff --git a/android/festago/domain/src/main/java/com/festago/festago/domain/repository/RecentSearchRepository.kt b/android/festago/domain/src/main/java/com/festago/festago/domain/repository/RecentSearchRepository.kt new file mode 100644 index 000000000..b24c8c6b8 --- /dev/null +++ b/android/festago/domain/src/main/java/com/festago/festago/domain/repository/RecentSearchRepository.kt @@ -0,0 +1,10 @@ +package com.festago.festago.domain.repository + +import com.festago.festago.domain.model.recentsearch.RecentSearchQuery + +interface RecentSearchRepository { + suspend fun insertOrReplaceRecentSearch(searchQuery: String) + suspend fun deleteRecentSearch(searchQuery: String) + suspend fun getRecentSearchQueries(limit: Int): List + suspend fun clearRecentSearches() +} diff --git a/android/festago/domain/src/main/java/com/festago/festago/domain/repository/SearchRepository.kt b/android/festago/domain/src/main/java/com/festago/festago/domain/repository/SearchRepository.kt new file mode 100644 index 000000000..b2129f848 --- /dev/null +++ b/android/festago/domain/src/main/java/com/festago/festago/domain/repository/SearchRepository.kt @@ -0,0 +1,11 @@ +package com.festago.festago.domain.repository + +import com.festago.festago.domain.model.festival.Festival +import com.festago.festago.domain.model.search.ArtistSearch +import com.festago.festago.domain.model.search.SchoolSearch + +interface SearchRepository { + suspend fun searchFestivals(searchQuery: String): Result> + suspend fun searchArtists(searchQuery: String): Result> + suspend fun searchSchools(searchQuery: String): Result> +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/customview/ClearEditText.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/customview/ClearEditText.kt new file mode 100644 index 000000000..2df1be761 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/customview/ClearEditText.kt @@ -0,0 +1,92 @@ +package com.festago.festago.presentation.ui.customview + +import android.content.Context +import android.graphics.drawable.Drawable +import android.text.Editable +import android.text.TextWatcher +import android.util.AttributeSet +import android.view.MotionEvent +import android.view.View +import android.view.View.OnFocusChangeListener +import android.view.View.OnTouchListener +import android.view.inputmethod.InputMethodManager +import androidx.appcompat.widget.AppCompatEditText +import androidx.core.content.ContextCompat +import androidx.core.graphics.drawable.DrawableCompat +import com.festago.festago.presentation.R + +class ClearEditText(context: Context, attrs: AttributeSet) : + AppCompatEditText(context, attrs), + TextWatcher, + OnTouchListener, + OnFocusChangeListener { + + private val clearDrawable: Drawable by lazy { + val drawable = ContextCompat.getDrawable(context, R.drawable.ic_circle_close)!! + DrawableCompat.wrap(drawable) + } + private var onFocusChangeListener: OnFocusChangeListener? = null + private var onTouchListener: OnTouchListener? = null + + init { + clearDrawable.setBounds( + 0, + 0, + clearDrawable.intrinsicWidth, + clearDrawable.intrinsicHeight, + ) + setClearIconVisible(false) + super.setOnTouchListener(this) + super.setOnFocusChangeListener(this) + addTextChangedListener(this) + } + + override fun setOnFocusChangeListener(onFocusChangeListener: OnFocusChangeListener) { + this.onFocusChangeListener = onFocusChangeListener + } + + override fun setOnTouchListener(onTouchListener: OnTouchListener) { + this.onTouchListener = onTouchListener + } + + override fun onFocusChange(view: View, hasFocus: Boolean) { + setClearIconVisible(text!!.isNotEmpty()) + if (onFocusChangeListener != null) { + onFocusChangeListener!!.onFocusChange(view, hasFocus) + } + } + + override fun onTouch(view: View, motionEvent: MotionEvent): Boolean { + val x = motionEvent.x.toInt() + if (clearDrawable.isVisible && x > width - paddingRight - clearDrawable.intrinsicWidth) { + if (motionEvent.action == MotionEvent.ACTION_UP) { + error = null + text = null + isFocusableInTouchMode = true + val inputMethodManager = context?.getSystemService(Context.INPUT_METHOD_SERVICE) as InputMethodManager + requestFocus() + inputMethodManager.showSoftInput(this, InputMethodManager.SHOW_IMPLICIT) + } + return true + } + return if (onTouchListener != null) { + onTouchListener!!.onTouch(view, motionEvent) + } else { + false + } + } + + override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) { + if (!isFocused) return + setClearIconVisible(s.isNotEmpty()) + } + + override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) = Unit + + override fun afterTextChanged(s: Editable) = Unit + + private fun setClearIconVisible(visible: Boolean) { + clearDrawable.setVisible(visible, false) + setCompoundDrawables(null, null, if (visible) clearDrawable else null, null) + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/customview/FestagoButton.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/customview/FestagoButton.kt new file mode 100644 index 000000000..af1318081 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/customview/FestagoButton.kt @@ -0,0 +1,27 @@ +package com.festago.festago.presentation.ui.customview + +import android.content.Context +import android.util.AttributeSet +import android.view.LayoutInflater +import androidx.constraintlayout.widget.ConstraintLayout +import com.festago.festago.presentation.R +import com.festago.festago.presentation.databinding.BtnFestagoBinding + +class FestagoButton(context: Context, attrs: AttributeSet) : + ConstraintLayout(context, attrs) { + + private val binding by lazy { + BtnFestagoBinding.inflate(LayoutInflater.from(context), this, true) + } + + init { + initAttrs(attrs) + } + + private fun initAttrs(attrs: AttributeSet) { + val typedArray = context.obtainStyledAttributes(attrs, R.styleable.FestagoButton) + val title = typedArray.getString(R.styleable.FestagoButton_title) + binding.tvFestagoBtn.text = title + typedArray.recycle() + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/home/HomeViewModel.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/home/HomeViewModel.kt index bdc2b89d2..06763528a 100644 --- a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/home/HomeViewModel.kt +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/home/HomeViewModel.kt @@ -1,5 +1,8 @@ package com.festago.festago.presentation.ui.home import androidx.lifecycle.ViewModel +import dagger.hilt.android.lifecycle.HiltViewModel +import javax.inject.Inject -class HomeViewModel : ViewModel() +@HiltViewModel +class HomeViewModel @Inject constructor() : ViewModel() diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/home/festivallist/FestivalListFragment.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/home/festivallist/FestivalListFragment.kt index 1f69952b8..edb26a588 100644 --- a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/home/festivallist/FestivalListFragment.kt +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/home/festivallist/FestivalListFragment.kt @@ -14,7 +14,7 @@ import androidx.navigation.fragment.findNavController import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.RecyclerView import com.festago.festago.presentation.databinding.FragmentFestivalListBinding -import com.festago.festago.presentation.ui.home.festivallist.FestivalListFragmentDirections.actionFestivalListFragmentToSchoolDetailFragment +import com.festago.festago.presentation.ui.home.festivallist.FestivalListFragmentDirections.actionFestivalListFragmentToSearchFragment import com.festago.festago.presentation.ui.home.festivallist.festival.FestivalListAdapter import com.festago.festago.presentation.ui.home.festivallist.uistate.FestivalListUiState import com.festago.festago.presentation.ui.home.festivallist.uistate.FestivalMoreItemUiState @@ -188,7 +188,7 @@ class FestivalListFragment : Fragment() { } private fun showSchoolDetail() { - findNavController().navigate(actionFestivalListFragmentToSchoolDetailFragment(0)) + findNavController().navigate(actionFestivalListFragmentToSearchFragment()) } private fun showNotificationList() { diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/SearchEvent.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/SearchEvent.kt new file mode 100644 index 000000000..535934c0f --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/SearchEvent.kt @@ -0,0 +1,9 @@ +package com.festago.festago.presentation.ui.search + +sealed interface SearchEvent { + class ShowFestivalDetail(val festivalId: Long) : SearchEvent + class ShowArtistDetail(val artistId: Long) : SearchEvent + class ShowSchoolDetail(val schoolId: Long) : SearchEvent + class UpdateSearchQuery(val searchQuery: String) : SearchEvent + object SearchBlank : SearchEvent +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/SearchFragment.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/SearchFragment.kt new file mode 100644 index 000000000..069511de6 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/SearchFragment.kt @@ -0,0 +1,225 @@ +package com.festago.festago.presentation.ui.search + +import android.content.Context +import android.content.Intent +import android.net.Uri +import android.os.Bundle +import android.view.KeyEvent +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import android.view.inputmethod.InputMethodManager +import android.widget.Toast +import androidx.core.view.WindowInsetsCompat +import androidx.core.view.doOnNextLayout +import androidx.fragment.app.Fragment +import androidx.fragment.app.viewModels +import androidx.navigation.fragment.findNavController +import com.festago.festago.presentation.R +import com.festago.festago.presentation.databinding.FragmentSearchBinding +import com.festago.festago.presentation.ui.search.recentsearch.RecentSearchAdapter +import com.festago.festago.presentation.ui.search.screen.ItemSearchScreenUiState +import com.festago.festago.presentation.ui.search.screen.ItemSearchScreenUiState.ArtistSearchScreen +import com.festago.festago.presentation.ui.search.screen.ItemSearchScreenUiState.FestivalSearchScreen +import com.festago.festago.presentation.ui.search.screen.ItemSearchScreenUiState.SchoolSearchScreen +import com.festago.festago.presentation.ui.search.screen.SearchScreenAdapter +import com.festago.festago.presentation.ui.search.uistate.SearchUiState +import com.festago.festago.presentation.util.repeatOnStarted +import com.festago.festago.presentation.util.setOnApplyWindowInsetsCompatListener +import com.google.android.material.tabs.TabLayoutMediator +import dagger.hilt.android.AndroidEntryPoint + +@AndroidEntryPoint +class SearchFragment : Fragment() { + + private var _binding: FragmentSearchBinding? = null + private val binding get() = _binding!! + + private val vm: SearchViewModel by viewModels() + + private lateinit var recentSearchAdapter: RecentSearchAdapter + private lateinit var searchScreenAdapter: SearchScreenAdapter + + override fun onCreateView( + inflater: LayoutInflater, + container: ViewGroup?, + savedInstanceState: Bundle?, + ): View { + _binding = FragmentSearchBinding.inflate(inflater) + binding.root.setOnApplyWindowInsetsCompatListener { view, windowInsets -> + val statusBarInsets = windowInsets.getInsets(WindowInsetsCompat.Type.statusBars()) + view.setPadding(0, statusBarInsets.top, 0, 0) + windowInsets + } + return binding.root + } + + override fun onViewCreated(view: View, savedInstanceState: Bundle?) { + super.onViewCreated(view, savedInstanceState) + initObserve() + initView() + } + + private fun initObserve() { + repeatOnStarted(viewLifecycleOwner) { + vm.uiState.collect { + binding.uiState = it + updateUi(it) + } + } + repeatOnStarted(viewLifecycleOwner) { + vm.event.collect { + handleEvent(it) + } + } + } + + private fun updateUi(uiState: SearchUiState) { + when (uiState) { + is SearchUiState.Loading, + is SearchUiState.Error, + -> Unit + + is SearchUiState.RecentSearchSuccess -> handleRecentSearchSuccess(uiState) + is SearchUiState.SearchSuccess -> handleSuccessSearch(uiState) + } + } + + private fun initView() { + initRecyclerView() + initBack() + initSearch() + initDeleteAll() + initViewPager() + } + + private fun initViewPager() { + searchScreenAdapter = SearchScreenAdapter() + binding.vpSearch.adapter = searchScreenAdapter + TabLayoutMediator(binding.tlSearch, binding.vpSearch) { tab, position -> + tab.text = when (position) { + ItemSearchScreenUiState.FESTIVAL_POSITION -> getString(R.string.search_tl_tab_festival) + ItemSearchScreenUiState.ARTIST_POSITION -> getString(R.string.search_tl_tab_Artist) + ItemSearchScreenUiState.SCHOOL_POSITION -> getString(R.string.search_tl_tab_school) + else -> "" + } + tab.view.setOnClickListener { + val currentScreen = when (tab.position) { + ItemSearchScreenUiState.FESTIVAL_POSITION -> FestivalSearchScreen(listOf()) + ItemSearchScreenUiState.ARTIST_POSITION -> ArtistSearchScreen(listOf()) + ItemSearchScreenUiState.SCHOOL_POSITION -> SchoolSearchScreen(listOf()) + else -> FestivalSearchScreen(listOf()) + } + vm.currentScreen = currentScreen + } + }.attach() + } + + private fun initRecyclerView() { + recentSearchAdapter = RecentSearchAdapter() + binding.rvRecentSearch.adapter = recentSearchAdapter + } + + private fun initBack() { + binding.ivBack.setOnClickListener { + requireActivity().onBackPressedDispatcher.onBackPressed() + } + } + + private fun initDeleteAll() { + binding.tvDeleteAll.setOnClickListener { + vm.deleteAllRecentSearch() + } + } + + private fun initSearch() { + binding.etSearch.setOnKeyListener { editText, keyCode, event -> + if ((event.action == KeyEvent.ACTION_DOWN) && (keyCode == KeyEvent.KEYCODE_ENTER)) { + vm.search(binding.etSearch.text.toString()) + val inputMethodManager = + context?.getSystemService(Context.INPUT_METHOD_SERVICE) as InputMethodManager + inputMethodManager.hideSoftInputFromWindow(editText.windowToken, 0) + true + } else { + false + } + } + vm.loadRecentSearch() + } + + private fun handleRecentSearchSuccess(uiState: SearchUiState.RecentSearchSuccess) { + recentSearchAdapter.submitList(uiState.recentSearchQueries) + } + + private fun handleSuccessSearch(uiState: SearchUiState.SearchSuccess) { + searchScreenAdapter.submitList( + listOf( + FestivalSearchScreen(uiState.searchedFestivals, ::requestAddSearchQuery), + ArtistSearchScreen(uiState.searchedArtists, ::requestAddSearchQuery), + SchoolSearchScreen(uiState.searchedSchools, ::requestAddSearchQuery), + ), + ) + initSearchTab() + } + + private fun initSearchTab() { + binding.vpSearch.doOnNextLayout { + binding.vpSearch.setCurrentItem(vm.currentScreen.screenPosition, false) + } + } + + private fun handleEvent(event: SearchEvent) { + when (event) { + is SearchEvent.ShowFestivalDetail -> { + findNavController().navigate( + SearchFragmentDirections.actionSearchFragmentToFestivalDetailFragment( + event.festivalId, + ), + ) + } + + is SearchEvent.ShowArtistDetail -> { + findNavController().navigate( + SearchFragmentDirections.actionSearchFragmentToArtistDetailFragment( + event.artistId, + ), + ) + } + + is SearchEvent.ShowSchoolDetail -> { + findNavController().navigate( + SearchFragmentDirections.actionSearchFragmentToSchoolDetailFragment( + event.schoolId, + ), + ) + } + + is SearchEvent.SearchBlank -> { + Toast.makeText( + requireContext(), + getString(R.string.search_cant_search_blank), + Toast.LENGTH_SHORT, + ).show() + } + + is SearchEvent.UpdateSearchQuery -> { + binding.etSearch.setText(event.searchQuery) + } + } + } + + private fun requestAddSearchQuery() { + startBrowser("https://forms.gle/y17dmCFw1jAYLR9H7") + } + + private fun startBrowser(url: String) { + val intent = Intent(Intent.ACTION_VIEW) + intent.data = Uri.parse(url) + startActivity(intent) + } + + override fun onDestroyView() { + _binding = null + super.onDestroyView() + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/SearchViewModel.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/SearchViewModel.kt new file mode 100644 index 000000000..58e02f39d --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/SearchViewModel.kt @@ -0,0 +1,170 @@ +package com.festago.festago.presentation.ui.search + +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import com.festago.festago.common.analytics.AnalyticsHelper +import com.festago.festago.common.analytics.logNetworkFailure +import com.festago.festago.domain.model.festival.Festival +import com.festago.festago.domain.model.recentsearch.RecentSearchQuery +import com.festago.festago.domain.model.search.ArtistSearch +import com.festago.festago.domain.model.search.SchoolSearch +import com.festago.festago.domain.repository.RecentSearchRepository +import com.festago.festago.domain.repository.SearchRepository +import com.festago.festago.presentation.ui.search.screen.ItemSearchScreenUiState +import com.festago.festago.presentation.ui.search.screen.ItemSearchScreenUiState.FestivalSearchScreen +import com.festago.festago.presentation.ui.search.uistate.ArtistSearchItemUiState +import com.festago.festago.presentation.ui.search.uistate.ArtistUiState +import com.festago.festago.presentation.ui.search.uistate.FestivalSearchItemUiState +import com.festago.festago.presentation.ui.search.uistate.RecentSearchItemUiState +import com.festago.festago.presentation.ui.search.uistate.SchoolSearchItemUiState +import com.festago.festago.presentation.ui.search.uistate.SchoolUiState +import com.festago.festago.presentation.ui.search.uistate.SearchUiState +import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.async +import kotlinx.coroutines.flow.MutableSharedFlow +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.SharedFlow +import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.flow.asSharedFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.launch +import javax.inject.Inject + +@HiltViewModel +class SearchViewModel @Inject constructor( + private val recentSearchRepository: RecentSearchRepository, + private val searchRepository: SearchRepository, + private val analyticsHelper: AnalyticsHelper, +) : ViewModel() { + + private val _uiState: MutableStateFlow = + MutableStateFlow(SearchUiState.RecentSearchSuccess(listOf())) + val uiState: StateFlow = _uiState.asStateFlow() + + private val _event = MutableSharedFlow() + val event: SharedFlow = _event.asSharedFlow() + + var currentScreen: ItemSearchScreenUiState = FestivalSearchScreen(listOf()) + + fun loadRecentSearch() { + if (uiState.value is SearchUiState.SearchSuccess) return + viewModelScope.launch { + _uiState.value = SearchUiState.RecentSearchSuccess( + recentSearchRepository.getRecentSearchQueries(10).map { it.toUiState() }, + ) + } + } + + fun search(searchQuery: String) { + viewModelScope.launch { + if (searchQuery == "") { + _event.emit(SearchEvent.SearchBlank) + return@launch + } + _uiState.value = SearchUiState.Loading + recentSearchRepository.insertOrReplaceRecentSearch(searchQuery) + val deferredFestivals = async { searchRepository.searchFestivals(searchQuery) } + val deferredArtist = async { searchRepository.searchArtists(searchQuery) } + val deferredSchools = async { searchRepository.searchSchools(searchQuery) } + runCatching { + _uiState.value = SearchUiState.SearchSuccess( + deferredFestivals.await().getOrThrow().map { it.toUiState() }, + deferredArtist.await().getOrThrow().map { it.toUiState() }, + deferredSchools.await().getOrThrow().map { it.toUiState() }, + ) + }.onFailure { + _uiState.value = SearchUiState.Error + analyticsHelper.logNetworkFailure( + key = KEY_SEARCH, + value = it.message.toString(), + ) + } + } + } + + private fun deleteRecentSearch(searchQuery: String) { + viewModelScope.launch { + recentSearchRepository.deleteRecentSearch(searchQuery) + loadRecentSearch() + } + } + + fun deleteAllRecentSearch() { + viewModelScope.launch { + recentSearchRepository.clearRecentSearches() + loadRecentSearch() + } + } + + private fun RecentSearchQuery.toUiState() = RecentSearchItemUiState( + recentQuery = query, + onQuerySearched = ::searchRecentQuery, + onRecentSearchDeleted = ::deleteRecentSearch, + ) + + private fun Festival.toUiState() = FestivalSearchItemUiState( + id = id, + name = name, + startDate = startDate, + endDate = endDate, + imageUrl = imageUrl, + schoolUiState = SchoolUiState( + id = school.id, + name = school.name, + ), + artists = artists.map { artist -> + ArtistUiState(artist.id, artist.name, artist.imageUrl, ::showArtistDetail) + }, + ::showFestivalDetail, + ) + + private fun ArtistSearch.toUiState() = ArtistSearchItemUiState( + id = id, + name = name, + profileImageUrl = profileImageUrl, + todayStage = todayStage, + upcomingStage = upcomingStage, + onArtistDetailClick = ::showArtistDetail, + ) + + private fun SchoolSearch.toUiState() = SchoolSearchItemUiState( + id = id, + name = name, + logoUrl = logoUrl, + upcomingFestivalStartDate = upcomingFestivalStartDate, + onSchoolSearchClick = ::showSchoolDetail, + ) + + private fun showFestivalDetail(festivalId: Long) { + viewModelScope.launch { + _event.emit(SearchEvent.ShowFestivalDetail(festivalId)) + } + } + + private fun showArtistDetail(artistId: Long) { + viewModelScope.launch { + _event.emit(SearchEvent.ShowArtistDetail(artistId)) + } + } + + private fun showSchoolDetail(schoolId: Long) { + viewModelScope.launch { + _event.emit(SearchEvent.ShowSchoolDetail(schoolId)) + } + } + + private fun searchRecentQuery(searchQuery: String) { + search(searchQuery) + updateSearchQuery(searchQuery) + } + + private fun updateSearchQuery(searchQuery: String) { + viewModelScope.launch { + _event.emit(SearchEvent.UpdateSearchQuery(searchQuery)) + } + } + + companion object { + private const val KEY_SEARCH = "KEY_SEARCH" + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/artistsearch/ArtistSearchAdapter.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/artistsearch/ArtistSearchAdapter.kt new file mode 100644 index 000000000..49e677853 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/artistsearch/ArtistSearchAdapter.kt @@ -0,0 +1,32 @@ +package com.festago.festago.presentation.ui.search.artistsearch + +import android.view.ViewGroup +import androidx.recyclerview.widget.DiffUtil +import androidx.recyclerview.widget.ListAdapter +import com.festago.festago.presentation.ui.search.uistate.ArtistSearchItemUiState + +class ArtistSearchAdapter : ListAdapter(diffUtil) { + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ArtistSearchViewHolder { + return ArtistSearchViewHolder.of(parent) + } + + override fun onBindViewHolder(holder: ArtistSearchViewHolder, position: Int) { + holder.bind(getItem(position)) + } + + companion object { + + val diffUtil = object : DiffUtil.ItemCallback() { + override fun areItemsTheSame( + oldItem: ArtistSearchItemUiState, + newItem: ArtistSearchItemUiState, + ): Boolean = oldItem.id == newItem.id + + override fun areContentsTheSame( + oldItem: ArtistSearchItemUiState, + newItem: ArtistSearchItemUiState, + ): Boolean = oldItem == newItem + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/artistsearch/ArtistSearchViewHolder.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/artistsearch/ArtistSearchViewHolder.kt new file mode 100644 index 000000000..5f74472e8 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/artistsearch/ArtistSearchViewHolder.kt @@ -0,0 +1,63 @@ +package com.festago.festago.presentation.ui.search.artistsearch + +import android.text.Spannable +import android.text.SpannableString +import android.text.style.ForegroundColorSpan +import android.view.LayoutInflater +import android.view.ViewGroup +import android.widget.TextView +import androidx.annotation.ColorInt +import androidx.annotation.StringRes +import androidx.recyclerview.widget.RecyclerView +import com.festago.festago.presentation.R +import com.festago.festago.presentation.databinding.ItemSearchArtistBinding +import com.festago.festago.presentation.ui.search.uistate.ArtistSearchItemUiState + +class ArtistSearchViewHolder( + private val binding: ItemSearchArtistBinding, +) : RecyclerView.ViewHolder(binding.root) { + + fun bind(item: ArtistSearchItemUiState) { + binding.item = item + binding.tvTodayStageCount.setStageCount( + count = item.todayStage, + stringRes = R.string.search_artist_tv_today_stage_count, + ) + binding.tvUpcomingStageCount.setStageCount( + item.upcomingStage, + stringRes = R.string.search_artist_tv_upcoming_stage_count, + ) + } + + private fun TextView.setStageCount(count: Int, @StringRes stringRes: Int) { + val stageCountText = context.getString(stringRes, count) + text = SpannableString(stageCountText).apply { + getPartialColorText( + start = COLOR_INDEX, + end = COLOR_INDEX + 1, + color = context.getColor(R.color.secondary_pink_01), + ) + } + } + + private fun SpannableString.getPartialColorText( + start: Int, + end: Int, + @ColorInt color: Int, + ) { + setSpan(ForegroundColorSpan(color), start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE) + } + + companion object { + private const val COLOR_INDEX = 6 + + fun of(parent: ViewGroup): ArtistSearchViewHolder { + val binding = ItemSearchArtistBinding.inflate( + LayoutInflater.from(parent.context), + parent, + false, + ) + return ArtistSearchViewHolder(binding) + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/FestivalSearchAdapter.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/FestivalSearchAdapter.kt new file mode 100644 index 000000000..bcc748f03 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/FestivalSearchAdapter.kt @@ -0,0 +1,33 @@ +package com.festago.festago.presentation.ui.search.festivalsearch + +import android.view.ViewGroup +import androidx.recyclerview.widget.DiffUtil +import androidx.recyclerview.widget.ListAdapter +import com.festago.festago.presentation.ui.search.uistate.FestivalSearchItemUiState + +class FestivalSearchAdapter : + ListAdapter(diffUtil) { + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): FestivalSearchViewHolder { + return FestivalSearchViewHolder.of(parent) + } + + override fun onBindViewHolder(holder: FestivalSearchViewHolder, position: Int) { + holder.bind(getItem(position)) + } + + companion object { + + val diffUtil = object : DiffUtil.ItemCallback() { + override fun areItemsTheSame( + oldItem: FestivalSearchItemUiState, + newItem: FestivalSearchItemUiState, + ): Boolean = oldItem.id == newItem.id + + override fun areContentsTheSame( + oldItem: FestivalSearchItemUiState, + newItem: FestivalSearchItemUiState, + ): Boolean = oldItem == newItem + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/FestivalSearchViewHolder.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/FestivalSearchViewHolder.kt new file mode 100644 index 000000000..bf60e0712 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/FestivalSearchViewHolder.kt @@ -0,0 +1,91 @@ +package com.festago.festago.presentation.ui.search.festivalsearch + +import android.content.res.Resources +import android.graphics.Rect +import android.util.TypedValue +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import android.widget.TextView +import androidx.appcompat.content.res.AppCompatResources +import androidx.recyclerview.widget.RecyclerView +import com.festago.festago.presentation.R +import com.festago.festago.presentation.databinding.ItemSearchFestivalBinding +import com.festago.festago.presentation.ui.search.festivalsearch.artist.ArtistAdapter +import com.festago.festago.presentation.ui.search.uistate.FestivalSearchItemUiState +import java.time.LocalDate + +class FestivalSearchViewHolder( + private val binding: ItemSearchFestivalBinding, +) : RecyclerView.ViewHolder(binding.root) { + + private val artistAdapter = ArtistAdapter() + + init { + binding.rvFestivalArtists.adapter = artistAdapter + binding.rvFestivalArtists.addItemDecoration(ArtistItemDecoration()) + } + + fun bind(item: FestivalSearchItemUiState) { + binding.item = item + artistAdapter.submitList(item.artists) + binding.tvFestivalDDay.bindFestivalDday(item) + } + + private fun TextView.bindFestivalDday(item: FestivalSearchItemUiState) { + when { + LocalDate.now() > item.endDate -> Unit + + LocalDate.now() >= item.startDate -> { + text = context.getString(R.string.festival_list_tv_dday_in_progress) + setTextColor(context.getColor(R.color.secondary_pink_01)) + background = AppCompatResources.getDrawable( + context, + R.drawable.bg_festival_list_dday_in_progress, + ) + } + + else -> { + val dDay = LocalDate.now().toEpochDay() - item.startDate.toEpochDay() + val backgroundColor = if (dDay >= -7L) { + context.getColor(R.color.secondary_pink_01) + } else { + context.getColor(R.color.contents_gray_07) + } + setBackgroundColor(backgroundColor) + setTextColor(context.getColor(R.color.background_gray_01)) + text = context.getString(R.string.festival_detail_tv_dday_format, dDay.toString()) + } + } + } + + private class ArtistItemDecoration : RecyclerView.ItemDecoration() { + override fun getItemOffsets( + outRect: Rect, + view: View, + parent: RecyclerView, + state: RecyclerView.State, + ) { + super.getItemOffsets(outRect, view, parent, state) + outRect.right = 8.dpToPx + } + + private val Int.dpToPx: Int + get() = TypedValue.applyDimension( + TypedValue.COMPLEX_UNIT_DIP, + this.toFloat(), + Resources.getSystem().displayMetrics, + ).toInt() + } + + companion object { + fun of(parent: ViewGroup): FestivalSearchViewHolder { + val binding = ItemSearchFestivalBinding.inflate( + LayoutInflater.from(parent.context), + parent, + false, + ) + return FestivalSearchViewHolder(binding) + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/artist/ArtistAdapter.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/artist/ArtistAdapter.kt new file mode 100644 index 000000000..103758f3f --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/artist/ArtistAdapter.kt @@ -0,0 +1,35 @@ +package com.festago.festago.presentation.ui.search.festivalsearch.artist + +import android.view.ViewGroup +import androidx.recyclerview.widget.DiffUtil +import androidx.recyclerview.widget.ListAdapter +import com.festago.festago.presentation.ui.search.uistate.ArtistUiState + +class ArtistAdapter : ListAdapter(diffUtil) { + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ArtistViewHolder { + return ArtistViewHolder.of(parent) + } + + override fun onBindViewHolder(holder: ArtistViewHolder, position: Int) { + holder.bind(getItem(position)) + } + + companion object { + private val diffUtil = object : DiffUtil.ItemCallback() { + override fun areItemsTheSame( + oldItem: ArtistUiState, + newItem: ArtistUiState, + ): Boolean { + return oldItem.id == newItem.id + } + + override fun areContentsTheSame( + oldItem: ArtistUiState, + newItem: ArtistUiState, + ): Boolean { + return oldItem == newItem + } + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/artist/ArtistViewHolder.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/artist/ArtistViewHolder.kt new file mode 100644 index 000000000..cb3f878a4 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/festivalsearch/artist/ArtistViewHolder.kt @@ -0,0 +1,27 @@ +package com.festago.festago.presentation.ui.search.festivalsearch.artist + +import android.view.LayoutInflater +import android.view.ViewGroup +import androidx.recyclerview.widget.RecyclerView +import com.festago.festago.presentation.databinding.ItemSearchFestivalArtistBinding +import com.festago.festago.presentation.ui.search.uistate.ArtistUiState + +class ArtistViewHolder( + private val binding: ItemSearchFestivalArtistBinding, +) : RecyclerView.ViewHolder(binding.root) { + + fun bind(item: ArtistUiState) { + binding.item = item + } + + companion object { + fun of(parent: ViewGroup): ArtistViewHolder { + val binding = ItemSearchFestivalArtistBinding.inflate( + LayoutInflater.from(parent.context), + parent, + false, + ) + return ArtistViewHolder(binding) + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/recentsearch/RecentSearchAdapter.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/recentsearch/RecentSearchAdapter.kt new file mode 100644 index 000000000..03721096e --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/recentsearch/RecentSearchAdapter.kt @@ -0,0 +1,36 @@ +package com.festago.festago.presentation.ui.search.recentsearch + +import android.view.ViewGroup +import androidx.recyclerview.widget.DiffUtil +import androidx.recyclerview.widget.ListAdapter +import com.festago.festago.presentation.ui.search.uistate.RecentSearchItemUiState + +class RecentSearchAdapter : ListAdapter(diffUtil) { + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecentSearchViewHolder { + return RecentSearchViewHolder.of(parent) + } + + override fun onBindViewHolder(holder: RecentSearchViewHolder, position: Int) { + holder.bind(getItem(position)) + } + + companion object { + + val diffUtil = object : DiffUtil.ItemCallback() { + override fun areItemsTheSame( + oldItem: RecentSearchItemUiState, + newItem: RecentSearchItemUiState, + ): Boolean { + return oldItem.recentQuery == newItem.recentQuery + } + + override fun areContentsTheSame( + oldItem: RecentSearchItemUiState, + newItem: RecentSearchItemUiState, + ): Boolean { + return oldItem == newItem + } + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/recentsearch/RecentSearchViewHolder.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/recentsearch/RecentSearchViewHolder.kt new file mode 100644 index 000000000..1f76330ca --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/recentsearch/RecentSearchViewHolder.kt @@ -0,0 +1,27 @@ +package com.festago.festago.presentation.ui.search.recentsearch + +import android.view.LayoutInflater +import android.view.ViewGroup +import androidx.recyclerview.widget.RecyclerView +import com.festago.festago.presentation.databinding.ItemRecentSearchBinding +import com.festago.festago.presentation.ui.search.uistate.RecentSearchItemUiState + +class RecentSearchViewHolder( + val binding: ItemRecentSearchBinding, +) : RecyclerView.ViewHolder(binding.root) { + + fun bind(item: RecentSearchItemUiState) { + binding.item = item + } + + companion object { + fun of(parent: ViewGroup): RecentSearchViewHolder { + val binding = ItemRecentSearchBinding.inflate( + LayoutInflater.from(parent.context), + parent, + false, + ) + return RecentSearchViewHolder(binding) + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/schoolSearchAdatper/SchoolSearchAdapter.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/schoolSearchAdatper/SchoolSearchAdapter.kt new file mode 100644 index 000000000..2735e3d71 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/schoolSearchAdatper/SchoolSearchAdapter.kt @@ -0,0 +1,32 @@ +package com.festago.festago.presentation.ui.search.schoolSearchAdatper + +import android.view.ViewGroup +import androidx.recyclerview.widget.DiffUtil +import androidx.recyclerview.widget.ListAdapter +import com.festago.festago.presentation.ui.search.uistate.SchoolSearchItemUiState + +class SchoolSearchAdapter : ListAdapter(diffUtil) { + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): SchoolSearchViewHolder { + return SchoolSearchViewHolder.of(parent) + } + + override fun onBindViewHolder(holder: SchoolSearchViewHolder, position: Int) { + holder.bind(getItem(position)) + } + + companion object { + + val diffUtil = object : DiffUtil.ItemCallback() { + override fun areItemsTheSame( + oldItem: SchoolSearchItemUiState, + newItem: SchoolSearchItemUiState, + ): Boolean = oldItem.id == newItem.id + + override fun areContentsTheSame( + oldItem: SchoolSearchItemUiState, + newItem: SchoolSearchItemUiState, + ): Boolean = oldItem == newItem + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/schoolSearchAdatper/SchoolSearchViewHolder.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/schoolSearchAdatper/SchoolSearchViewHolder.kt new file mode 100644 index 000000000..26d3b37e4 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/schoolSearchAdatper/SchoolSearchViewHolder.kt @@ -0,0 +1,56 @@ +package com.festago.festago.presentation.ui.search.schoolSearchAdatper + +import android.view.LayoutInflater +import android.view.ViewGroup +import android.widget.TextView +import androidx.recyclerview.widget.RecyclerView +import com.festago.festago.presentation.R +import com.festago.festago.presentation.databinding.ItemSearchSchoolBinding +import com.festago.festago.presentation.ui.search.uistate.SchoolSearchItemUiState +import java.time.LocalDate + +class SchoolSearchViewHolder( + private val binding: ItemSearchSchoolBinding, +) : RecyclerView.ViewHolder(binding.root) { + + fun bind(item: SchoolSearchItemUiState) { + binding.item = item + binding.tvSchoolFestivalDday.setSchoolFestivalDday(item.upcomingFestivalStartDate) + } + + private fun TextView.setSchoolFestivalDday( + upcomingFestivalStartDate: LocalDate?, + ) { + when { + upcomingFestivalStartDate == null -> { + text = context.getString(R.string.search_school_tv_no_plan) + setTextColor(context.getColor(R.color.contents_gray_05)) + } + + LocalDate.now() >= upcomingFestivalStartDate -> { + text = context.getString(R.string.search_school_tv_dday_in_progress) + setTextColor(context.getColor(R.color.secondary_pink_01)) + } + + LocalDate.now() < upcomingFestivalStartDate -> { + val dDay = + LocalDate.now().toEpochDay() - upcomingFestivalStartDate.toEpochDay() + text = context.getString(R.string.search_school_tv_dday_format, dDay.toString()) + val colorId = + if (dDay >= -7L) R.color.secondary_pink_01 else R.color.contents_gray_07 + setTextColor(context.getColor(colorId)) + } + } + } + + companion object { + fun of(parent: ViewGroup): SchoolSearchViewHolder { + val binding = ItemSearchSchoolBinding.inflate( + LayoutInflater.from(parent.context), + parent, + false, + ) + return SchoolSearchViewHolder(binding) + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/ArtistSearchScreenViewHolder.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/ArtistSearchScreenViewHolder.kt new file mode 100644 index 000000000..fe9233a74 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/ArtistSearchScreenViewHolder.kt @@ -0,0 +1,44 @@ +package com.festago.festago.presentation.ui.search.screen + +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import com.festago.festago.presentation.databinding.ItemArtistSearchScreenBinding +import com.festago.festago.presentation.ui.search.artistsearch.ArtistSearchAdapter + +class ArtistSearchScreenViewHolder( + private val binding: ItemArtistSearchScreenBinding, +) : SearchScreenViewHolder(binding) { + + private val artistSearchAdapter: ArtistSearchAdapter = ArtistSearchAdapter() + + init { + binding.rvArtists.adapter = artistSearchAdapter + } + + fun bind(item: ItemSearchScreenUiState.ArtistSearchScreen) { + artistSearchAdapter.submitList(item.artistSearches) + setNoResultVisibility(item) + binding.btnFestago.setOnClickListener { + item.onAddSearchQueryClick() + } + } + + private fun setNoResultVisibility(item: ItemSearchScreenUiState.ArtistSearchScreen) { + val visibility = if (item.artistSearches.isEmpty()) View.VISIBLE else View.GONE + binding.tvNoResult.visibility = visibility + binding.tvNoResultGuide.visibility = visibility + binding.btnFestago.visibility = visibility + } + + companion object { + fun of(parent: ViewGroup): ArtistSearchScreenViewHolder { + val binding = ItemArtistSearchScreenBinding.inflate( + LayoutInflater.from(parent.context), + parent, + false, + ) + return ArtistSearchScreenViewHolder(binding) + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/FestivalSearchScreenViewHolder.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/FestivalSearchScreenViewHolder.kt new file mode 100644 index 000000000..eb33d7c81 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/FestivalSearchScreenViewHolder.kt @@ -0,0 +1,44 @@ +package com.festago.festago.presentation.ui.search.screen + +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import com.festago.festago.presentation.databinding.ItemFestivalSearchScreenBinding +import com.festago.festago.presentation.ui.search.festivalsearch.FestivalSearchAdapter + +class FestivalSearchScreenViewHolder( + private val binding: ItemFestivalSearchScreenBinding, +) : SearchScreenViewHolder(binding) { + + private val festivalSearchAdapter: FestivalSearchAdapter = FestivalSearchAdapter() + + init { + binding.rvFestivals.adapter = festivalSearchAdapter + } + + fun bind(item: ItemSearchScreenUiState.FestivalSearchScreen) { + festivalSearchAdapter.submitList(item.festivalSearches) + setNoResultVisibility(item) + binding.btnFestago.setOnClickListener { + item.onAddSearchQueryClick() + } + } + + private fun setNoResultVisibility(item: ItemSearchScreenUiState.FestivalSearchScreen) { + val visibility = if (item.festivalSearches.isEmpty()) View.VISIBLE else View.GONE + binding.tvNoResult.visibility = visibility + binding.tvNoResultGuide.visibility = visibility + binding.btnFestago.visibility = visibility + } + + companion object { + fun of(parent: ViewGroup): FestivalSearchScreenViewHolder { + val binding = ItemFestivalSearchScreenBinding.inflate( + LayoutInflater.from(parent.context), + parent, + false, + ) + return FestivalSearchScreenViewHolder(binding) + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/ItemSearchScreenUiState.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/ItemSearchScreenUiState.kt new file mode 100644 index 000000000..78a886270 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/ItemSearchScreenUiState.kt @@ -0,0 +1,30 @@ +package com.festago.festago.presentation.ui.search.screen + +import com.festago.festago.presentation.ui.search.uistate.ArtistSearchItemUiState +import com.festago.festago.presentation.ui.search.uistate.FestivalSearchItemUiState +import com.festago.festago.presentation.ui.search.uistate.SchoolSearchItemUiState + +sealed class ItemSearchScreenUiState( + val screenPosition: Int, +) { + data class FestivalSearchScreen( + val festivalSearches: List, + val onAddSearchQueryClick: () -> Unit = {}, + ) : ItemSearchScreenUiState(FESTIVAL_POSITION) + + data class ArtistSearchScreen( + val artistSearches: List, + val onAddSearchQueryClick: () -> Unit = {}, + ) : ItemSearchScreenUiState(ARTIST_POSITION) + + data class SchoolSearchScreen( + val schoolSearches: List, + val onAddSearchQueryClick: () -> Unit = {}, + ) : ItemSearchScreenUiState(SCHOOL_POSITION) + + companion object { + const val FESTIVAL_POSITION = 0 + const val ARTIST_POSITION = 1 + const val SCHOOL_POSITION = 2 + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/SchoolSearchScreenViewHolder.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/SchoolSearchScreenViewHolder.kt new file mode 100644 index 000000000..8702fa879 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/SchoolSearchScreenViewHolder.kt @@ -0,0 +1,42 @@ +package com.festago.festago.presentation.ui.search.screen + +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import com.festago.festago.presentation.databinding.ItemSchoolSearchScreenBinding +import com.festago.festago.presentation.ui.search.schoolSearchAdatper.SchoolSearchAdapter + +class SchoolSearchScreenViewHolder(private val binding: ItemSchoolSearchScreenBinding) : + SearchScreenViewHolder(binding) { + private val schoolSearchAdapter: SchoolSearchAdapter = SchoolSearchAdapter() + + init { + binding.rvSchools.adapter = schoolSearchAdapter + } + + fun bind(item: ItemSearchScreenUiState.SchoolSearchScreen) { + schoolSearchAdapter.submitList(item.schoolSearches) + setNoResultVisibility(item) + binding.btnFestago.setOnClickListener { + item.onAddSearchQueryClick() + } + } + + private fun setNoResultVisibility(item: ItemSearchScreenUiState.SchoolSearchScreen) { + val visibility = if (item.schoolSearches.isEmpty()) View.VISIBLE else View.GONE + binding.tvNoResult.visibility = visibility + binding.tvNoResultGuide.visibility = visibility + binding.btnFestago.visibility = visibility + } + + companion object { + fun of(parent: ViewGroup): SchoolSearchScreenViewHolder { + val binding = ItemSchoolSearchScreenBinding.inflate( + LayoutInflater.from(parent.context), + parent, + false, + ) + return SchoolSearchScreenViewHolder(binding) + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/SearchScreenAdapter.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/SearchScreenAdapter.kt new file mode 100644 index 000000000..23c7781aa --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/SearchScreenAdapter.kt @@ -0,0 +1,47 @@ +package com.festago.festago.presentation.ui.search.screen + +import android.view.ViewGroup +import androidx.recyclerview.widget.DiffUtil +import androidx.recyclerview.widget.ListAdapter + +class SearchScreenAdapter : ListAdapter(diffUtil) { + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): SearchScreenViewHolder { + return when (viewType) { + 1 -> FestivalSearchScreenViewHolder.of(parent) + 2 -> ArtistSearchScreenViewHolder.of(parent) + 3 -> SchoolSearchScreenViewHolder.of(parent) + else -> throw IllegalArgumentException("Invalid viewType") + } + } + + override fun onBindViewHolder(holder: SearchScreenViewHolder, position: Int) { + val item = getItem(position) + return when (holder) { + is FestivalSearchScreenViewHolder -> holder.bind(item as ItemSearchScreenUiState.FestivalSearchScreen) + is ArtistSearchScreenViewHolder -> holder.bind(item as ItemSearchScreenUiState.ArtistSearchScreen) + is SchoolSearchScreenViewHolder -> holder.bind(item as ItemSearchScreenUiState.SchoolSearchScreen) + } + } + + override fun getItemViewType(position: Int): Int { + return when (getItem(position)) { + is ItemSearchScreenUiState.FestivalSearchScreen -> 1 + is ItemSearchScreenUiState.ArtistSearchScreen -> 2 + is ItemSearchScreenUiState.SchoolSearchScreen -> 3 + } + } + + companion object { + val diffUtil = object : DiffUtil.ItemCallback() { + override fun areItemsTheSame( + oldItem: ItemSearchScreenUiState, + newItem: ItemSearchScreenUiState, + ): Boolean = oldItem === newItem + + override fun areContentsTheSame( + oldItem: ItemSearchScreenUiState, + newItem: ItemSearchScreenUiState, + ): Boolean = oldItem == newItem + } + } +} diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/SearchScreenViewHolder.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/SearchScreenViewHolder.kt new file mode 100644 index 000000000..40df9c049 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/screen/SearchScreenViewHolder.kt @@ -0,0 +1,7 @@ +package com.festago.festago.presentation.ui.search.screen + +import androidx.databinding.ViewDataBinding +import androidx.recyclerview.widget.RecyclerView + +sealed class SearchScreenViewHolder(binding: ViewDataBinding) : + RecyclerView.ViewHolder(binding.root) diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/ArtistSearchItemUiState.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/ArtistSearchItemUiState.kt new file mode 100644 index 000000000..99eaf90e9 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/ArtistSearchItemUiState.kt @@ -0,0 +1,10 @@ +package com.festago.festago.presentation.ui.search.uistate + +data class ArtistSearchItemUiState( + val id: Long, + val name: String, + val profileImageUrl: String, + val todayStage: Int, + val upcomingStage: Int, + val onArtistDetailClick: (Long) -> Unit, +) diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/ArtistUiState.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/ArtistUiState.kt new file mode 100644 index 000000000..9d559f34b --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/ArtistUiState.kt @@ -0,0 +1,8 @@ +package com.festago.festago.presentation.ui.search.uistate + +data class ArtistUiState( + val id: Long, + val name: String, + val imageUrl: String, + val onArtistDetailClick: (Long) -> Unit, +) diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/FestivalSearchItemUiState.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/FestivalSearchItemUiState.kt new file mode 100644 index 000000000..218141b52 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/FestivalSearchItemUiState.kt @@ -0,0 +1,14 @@ +package com.festago.festago.presentation.ui.search.uistate + +import java.time.LocalDate + +data class FestivalSearchItemUiState( + val id: Long, + val name: String, + val startDate: LocalDate, + val endDate: LocalDate, + val imageUrl: String, + val schoolUiState: SchoolUiState, + val artists: List, + val onFestivalSearchClick: (festivalId: Long) -> Unit, +) diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/RecentSearchItemUiState.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/RecentSearchItemUiState.kt new file mode 100644 index 000000000..72ad05d82 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/RecentSearchItemUiState.kt @@ -0,0 +1,7 @@ +package com.festago.festago.presentation.ui.search.uistate + +data class RecentSearchItemUiState( + val recentQuery: String, + val onQuerySearched: (recentQuery: String) -> Unit, + val onRecentSearchDeleted: (recentQuery: String) -> Unit, +) diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/SchoolSearchItemUiState.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/SchoolSearchItemUiState.kt new file mode 100644 index 000000000..8b4a459fc --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/SchoolSearchItemUiState.kt @@ -0,0 +1,11 @@ +package com.festago.festago.presentation.ui.search.uistate + +import java.time.LocalDate + +data class SchoolSearchItemUiState( + val id: Long, + val name: String, + val logoUrl: String, + val upcomingFestivalStartDate: LocalDate?, + val onSchoolSearchClick: (schoolId: Long) -> Unit, +) diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/SchoolUiState.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/SchoolUiState.kt new file mode 100644 index 000000000..446acb2ba --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/SchoolUiState.kt @@ -0,0 +1,6 @@ +package com.festago.festago.presentation.ui.search.uistate + +data class SchoolUiState( + val id: Long, + val name: String, +) diff --git a/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/SearchUiState.kt b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/SearchUiState.kt new file mode 100644 index 000000000..d80d392f2 --- /dev/null +++ b/android/festago/presentation/src/main/java/com/festago/festago/presentation/ui/search/uistate/SearchUiState.kt @@ -0,0 +1,24 @@ +package com.festago.festago.presentation.ui.search.uistate + +sealed interface SearchUiState { + object Loading : SearchUiState + + data class RecentSearchSuccess( + val recentSearchQueries: List, + ) : SearchUiState + + data class SearchSuccess( + val searchedFestivals: List, + val searchedArtists: List, + val searchedSchools: List, + ) : SearchUiState + + object Error : SearchUiState + + val shouldShowNotEmptyRecentSearchSuccess get() = this is RecentSearchSuccess && recentSearchQueries.isNotEmpty() + val shouldShowEmptyRecentSearchSuccess get() = this is RecentSearchSuccess && recentSearchQueries.isEmpty() + val shouldShowRecentSearchSuccess get() = this is RecentSearchSuccess + val shouldShowSearchSuccess get() = this is SearchSuccess + val shouldShowLoading get() = this is Loading + val shouldShowError get() = this is Error +} diff --git a/android/festago/presentation/src/main/res/drawable/ic_arrow_back_ios_new_24.xml b/android/festago/presentation/src/main/res/drawable/ic_arrow_back_ios_new_24.xml deleted file mode 100644 index 6bd5650e1..000000000 --- a/android/festago/presentation/src/main/res/drawable/ic_arrow_back_ios_new_24.xml +++ /dev/null @@ -1,5 +0,0 @@ - - - diff --git a/android/festago/presentation/src/main/res/drawable/ic_back_gray_07.xml b/android/festago/presentation/src/main/res/drawable/ic_back_gray_07.xml new file mode 100644 index 000000000..58dfdf648 --- /dev/null +++ b/android/festago/presentation/src/main/res/drawable/ic_back_gray_07.xml @@ -0,0 +1,13 @@ + + + diff --git a/android/festago/presentation/src/main/res/drawable/ic_circle_close.xml b/android/festago/presentation/src/main/res/drawable/ic_circle_close.xml new file mode 100644 index 000000000..b12812144 --- /dev/null +++ b/android/festago/presentation/src/main/res/drawable/ic_circle_close.xml @@ -0,0 +1,13 @@ + + + diff --git a/android/festago/presentation/src/main/res/drawable/ic_close.xml b/android/festago/presentation/src/main/res/drawable/ic_close.xml new file mode 100644 index 000000000..03c1b023f --- /dev/null +++ b/android/festago/presentation/src/main/res/drawable/ic_close.xml @@ -0,0 +1,10 @@ + + + diff --git a/android/festago/presentation/src/main/res/drawable/ic_next.xml b/android/festago/presentation/src/main/res/drawable/ic_next.xml new file mode 100644 index 000000000..d9d292377 --- /dev/null +++ b/android/festago/presentation/src/main/res/drawable/ic_next.xml @@ -0,0 +1,13 @@ + + + diff --git a/android/festago/presentation/src/main/res/drawable/ic_x_close.xml b/android/festago/presentation/src/main/res/drawable/ic_x_close.xml new file mode 100644 index 000000000..0d1129d89 --- /dev/null +++ b/android/festago/presentation/src/main/res/drawable/ic_x_close.xml @@ -0,0 +1,13 @@ + + + diff --git a/android/festago/presentation/src/main/res/layout/activity_notification_list.xml b/android/festago/presentation/src/main/res/layout/activity_notification_list.xml index c637ebb1a..d132f9d45 100644 --- a/android/festago/presentation/src/main/res/layout/activity_notification_list.xml +++ b/android/festago/presentation/src/main/res/layout/activity_notification_list.xml @@ -21,10 +21,10 @@ + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/layout/fragment_artist_detail.xml b/android/festago/presentation/src/main/res/layout/fragment_artist_detail.xml index d2fe488db..9c9daa8b9 100644 --- a/android/festago/presentation/src/main/res/layout/fragment_artist_detail.xml +++ b/android/festago/presentation/src/main/res/layout/fragment_artist_detail.xml @@ -47,10 +47,10 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/layout/item_artist_search_screen.xml b/android/festago/presentation/src/main/res/layout/item_artist_search_screen.xml new file mode 100644 index 000000000..36e3037be --- /dev/null +++ b/android/festago/presentation/src/main/res/layout/item_artist_search_screen.xml @@ -0,0 +1,68 @@ + + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/layout/item_festival_search_screen.xml b/android/festago/presentation/src/main/res/layout/item_festival_search_screen.xml new file mode 100644 index 000000000..30893979e --- /dev/null +++ b/android/festago/presentation/src/main/res/layout/item_festival_search_screen.xml @@ -0,0 +1,69 @@ + + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/layout/item_recent_search.xml b/android/festago/presentation/src/main/res/layout/item_recent_search.xml new file mode 100644 index 000000000..dff693669 --- /dev/null +++ b/android/festago/presentation/src/main/res/layout/item_recent_search.xml @@ -0,0 +1,53 @@ + + + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/layout/item_school_search_screen.xml b/android/festago/presentation/src/main/res/layout/item_school_search_screen.xml new file mode 100644 index 000000000..bf5c0dc52 --- /dev/null +++ b/android/festago/presentation/src/main/res/layout/item_school_search_screen.xml @@ -0,0 +1,67 @@ + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/layout/item_search_artist.xml b/android/festago/presentation/src/main/res/layout/item_search_artist.xml new file mode 100644 index 000000000..30d024ad8 --- /dev/null +++ b/android/festago/presentation/src/main/res/layout/item_search_artist.xml @@ -0,0 +1,90 @@ + + + + + + + + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/layout/item_search_festival.xml b/android/festago/presentation/src/main/res/layout/item_search_festival.xml new file mode 100644 index 000000000..80f71338b --- /dev/null +++ b/android/festago/presentation/src/main/res/layout/item_search_festival.xml @@ -0,0 +1,107 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/layout/item_search_festival_artist.xml b/android/festago/presentation/src/main/res/layout/item_search_festival_artist.xml new file mode 100644 index 000000000..c1c402ca9 --- /dev/null +++ b/android/festago/presentation/src/main/res/layout/item_search_festival_artist.xml @@ -0,0 +1,56 @@ + + + + + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/layout/item_search_school.xml b/android/festago/presentation/src/main/res/layout/item_search_school.xml new file mode 100644 index 000000000..7652d366b --- /dev/null +++ b/android/festago/presentation/src/main/res/layout/item_search_school.xml @@ -0,0 +1,76 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/android/festago/presentation/src/main/res/navigation/festival_list.xml b/android/festago/presentation/src/main/res/navigation/festival_list.xml index 9df3fa8be..554a8d764 100644 --- a/android/festago/presentation/src/main/res/navigation/festival_list.xml +++ b/android/festago/presentation/src/main/res/navigation/festival_list.xml @@ -25,6 +25,11 @@ app:destination="@id/festivalDetailFragment" app:enterAnim="@android:anim/slide_in_left" app:popExitAnim="@android:anim/slide_out_right" /> + + + + + + + diff --git a/android/festago/presentation/src/main/res/values/attrs.xml b/android/festago/presentation/src/main/res/values/attrs.xml new file mode 100644 index 000000000..23b2df7d8 --- /dev/null +++ b/android/festago/presentation/src/main/res/values/attrs.xml @@ -0,0 +1,6 @@ + + + + + + diff --git a/android/festago/presentation/src/main/res/values/strings.xml b/android/festago/presentation/src/main/res/values/strings.xml index 544729f9a..9d27be328 100644 --- a/android/festago/presentation/src/main/res/values/strings.xml +++ b/android/festago/presentation/src/main/res/values/strings.xml @@ -33,4 +33,21 @@ D%1$s 종료 + + 학교명, 아티스트명, 축제명으로 입력하세요. + 최근 검색어 + 전체 삭제 + 최근 검색 내역이 없어요 + 축제 + 아티스트 + 학교 + 검색어를 입력해주세요 + 축제 중 + D%1$s + 예정 없음 + 오늘 공연 %d개 + 공연 예정 %d개 + 앗! 검색 결과가 없네요 + 찾으시는 정보가 없다면 저희에게 알려주세요 + 추가 요청하러 가기