feat(chat-character): 신규 캐릭터 전체보기 화면 및 API 연동 추가
This commit is contained in:
@@ -43,6 +43,14 @@ interface CharacterApi {
|
||||
@Query("size") size: Int
|
||||
): Single<ApiResponse<CharacterImageListResponse>>
|
||||
|
||||
// 신규 캐릭터 전체보기
|
||||
@GET("/api/chat/character/recent")
|
||||
fun getRecentCharacters(
|
||||
@Header("Authorization") authHeader: String,
|
||||
@Query("page") page: Int,
|
||||
@Query("size") size: Int
|
||||
): Single<ApiResponse<kr.co.vividnext.sodalive.chat.character.newcharacters.RecentCharactersResponse>>
|
||||
|
||||
@POST("/api/chat/character/image/purchase")
|
||||
fun purchaseCharacterImage(
|
||||
@Header("Authorization") authHeader: String,
|
||||
|
||||
@@ -22,6 +22,7 @@ import kr.co.vividnext.sodalive.base.SodaDialog
|
||||
import kr.co.vividnext.sodalive.chat.character.curation.CurationSectionAdapter
|
||||
import kr.co.vividnext.sodalive.chat.character.detail.CharacterDetailActivity
|
||||
import kr.co.vividnext.sodalive.chat.character.detail.CharacterDetailActivity.Companion.EXTRA_CHARACTER_ID
|
||||
import kr.co.vividnext.sodalive.chat.character.newcharacters.NewCharactersAllActivity
|
||||
import kr.co.vividnext.sodalive.chat.character.recent.RecentCharacterAdapter
|
||||
import kr.co.vividnext.sodalive.common.LoadingDialog
|
||||
import kr.co.vividnext.sodalive.common.SharedPreferenceManager
|
||||
@@ -289,6 +290,14 @@ class CharacterTabFragment : BaseFragment<FragmentCharacterTabBinding>(
|
||||
recyclerView.adapter = newCharacterAdapter
|
||||
|
||||
binding.tvNewCharacterAll.setOnClickListener {
|
||||
ensureLoginAndAuth {
|
||||
startActivity(
|
||||
Intent(
|
||||
requireContext(),
|
||||
NewCharactersAllActivity::class.java
|
||||
)
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
// 신규 캐릭터 LiveData 구독
|
||||
|
||||
@@ -0,0 +1,92 @@
|
||||
package kr.co.vividnext.sodalive.chat.character.newcharacters
|
||||
|
||||
import android.content.Intent
|
||||
import android.os.Bundle
|
||||
import android.view.View
|
||||
import androidx.recyclerview.widget.GridLayoutManager
|
||||
import androidx.recyclerview.widget.LinearLayoutManager
|
||||
import androidx.recyclerview.widget.RecyclerView
|
||||
import kr.co.vividnext.sodalive.base.BaseActivity
|
||||
import kr.co.vividnext.sodalive.chat.character.detail.CharacterDetailActivity
|
||||
import kr.co.vividnext.sodalive.chat.character.detail.CharacterDetailActivity.Companion.EXTRA_CHARACTER_ID
|
||||
import kr.co.vividnext.sodalive.common.GridSpacingItemDecoration
|
||||
import kr.co.vividnext.sodalive.common.LoadingDialog
|
||||
import kr.co.vividnext.sodalive.databinding.ActivityNewCharactersAllBinding
|
||||
import kr.co.vividnext.sodalive.extensions.dpToPx
|
||||
import org.koin.android.ext.android.inject
|
||||
|
||||
class NewCharactersAllActivity : BaseActivity<ActivityNewCharactersAllBinding>(
|
||||
ActivityNewCharactersAllBinding::inflate
|
||||
) {
|
||||
private val viewModel: NewCharactersAllViewModel by inject()
|
||||
|
||||
private lateinit var adapter: NewCharactersAllAdapter
|
||||
private lateinit var loadingDialog: LoadingDialog
|
||||
|
||||
override fun onCreate(savedInstanceState: Bundle?) {
|
||||
super.onCreate(savedInstanceState)
|
||||
setupView()
|
||||
bindData()
|
||||
viewModel.loadMore()
|
||||
}
|
||||
|
||||
override fun setupView() {
|
||||
loadingDialog = LoadingDialog(this, layoutInflater)
|
||||
binding.toolbar.tvBack.text = "신규 캐릭터 전체보기"
|
||||
binding.toolbar.tvBack.setOnClickListener { finish() }
|
||||
|
||||
val spanCount = 3
|
||||
val spacingPx = 8f.dpToPx().toInt()
|
||||
|
||||
adapter = NewCharactersAllAdapter { characterId ->
|
||||
startActivity(
|
||||
Intent(this, CharacterDetailActivity::class.java).apply {
|
||||
putExtra(EXTRA_CHARACTER_ID, characterId)
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
binding.rvCharacters.layoutManager = GridLayoutManager(this, spanCount)
|
||||
binding.rvCharacters.addItemDecoration(
|
||||
GridSpacingItemDecoration(
|
||||
spanCount,
|
||||
spacingPx,
|
||||
false
|
||||
)
|
||||
)
|
||||
binding.rvCharacters.addOnScrollListener(object : RecyclerView.OnScrollListener() {
|
||||
override fun onScrolled(recyclerView: RecyclerView, dx: Int, dy: Int) {
|
||||
super.onScrolled(recyclerView, dx, dy)
|
||||
val lastVisibleItemPosition = (recyclerView.layoutManager as LinearLayoutManager)
|
||||
.findLastVisibleItemPosition()
|
||||
val totalItemCount = recyclerView.adapter?.itemCount ?: 0
|
||||
if (
|
||||
!recyclerView.canScrollVertically(1) &&
|
||||
lastVisibleItemPosition >= totalItemCount - 1
|
||||
) {
|
||||
viewModel.loadMore()
|
||||
}
|
||||
}
|
||||
})
|
||||
binding.rvCharacters.adapter = adapter
|
||||
}
|
||||
|
||||
private fun bindData() {
|
||||
viewModel.isLoading.observe(this) { isLoading ->
|
||||
if (isLoading) loadingDialog.show(screenWidth) else loadingDialog.dismiss()
|
||||
}
|
||||
|
||||
viewModel.totalCount.observe(this) { count ->
|
||||
binding.tvTotalCount.text = "$count"
|
||||
}
|
||||
|
||||
viewModel.items.observe(this) { list ->
|
||||
adapter.addItems(list.drop(adapter.itemCount))
|
||||
binding.rvCharacters.visibility = if (list.isEmpty()) View.GONE else View.VISIBLE
|
||||
}
|
||||
|
||||
viewModel.toastLiveData.observe(this) { message ->
|
||||
message?.let { showToast(it) }
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,55 @@
|
||||
package kr.co.vividnext.sodalive.chat.character.newcharacters
|
||||
|
||||
import android.annotation.SuppressLint
|
||||
import android.view.LayoutInflater
|
||||
import android.view.ViewGroup
|
||||
import androidx.recyclerview.widget.RecyclerView
|
||||
import coil.load
|
||||
import coil.transform.RoundedCornersTransformation
|
||||
import kr.co.vividnext.sodalive.R
|
||||
import kr.co.vividnext.sodalive.chat.character.Character
|
||||
import kr.co.vividnext.sodalive.databinding.ItemNewCharacterAllBinding
|
||||
import kr.co.vividnext.sodalive.extensions.dpToPx
|
||||
|
||||
class NewCharactersAllAdapter(
|
||||
private val onClick: (Long) -> Unit
|
||||
) : RecyclerView.Adapter<NewCharactersAllAdapter.VH>() {
|
||||
|
||||
private val items = mutableListOf<Character>()
|
||||
|
||||
inner class VH(val binding: ItemNewCharacterAllBinding) : RecyclerView.ViewHolder(binding.root) {
|
||||
fun bind(item: Character) {
|
||||
binding.tvCharacterName.text = item.name
|
||||
binding.tvCharacterDescription.text = item.description
|
||||
binding.ivCharacter.load(item.imageUrl) {
|
||||
crossfade(true)
|
||||
placeholder(R.drawable.ic_logo_service_center)
|
||||
transformations(RoundedCornersTransformation(16f.dpToPx()))
|
||||
}
|
||||
binding.root.setOnClickListener { onClick(item.characterId) }
|
||||
}
|
||||
}
|
||||
|
||||
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): VH {
|
||||
val binding = ItemNewCharacterAllBinding.inflate(LayoutInflater.from(parent.context), parent, false)
|
||||
return VH(binding)
|
||||
}
|
||||
|
||||
override fun getItemCount(): Int = items.size
|
||||
|
||||
override fun onBindViewHolder(holder: VH, position: Int) {
|
||||
holder.bind(items[position])
|
||||
}
|
||||
|
||||
fun addItems(newItems: List<Character>) {
|
||||
val start = items.size
|
||||
items.addAll(newItems)
|
||||
notifyItemRangeInserted(start, newItems.size)
|
||||
}
|
||||
|
||||
@SuppressLint("NotifyDataSetChanged")
|
||||
fun clear() {
|
||||
items.clear()
|
||||
notifyDataSetChanged()
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,68 @@
|
||||
package kr.co.vividnext.sodalive.chat.character.newcharacters
|
||||
|
||||
import androidx.lifecycle.LiveData
|
||||
import androidx.lifecycle.MutableLiveData
|
||||
import com.orhanobut.logger.Logger
|
||||
import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers
|
||||
import io.reactivex.rxjava3.schedulers.Schedulers
|
||||
import kr.co.vividnext.sodalive.base.BaseViewModel
|
||||
import kr.co.vividnext.sodalive.chat.character.Character
|
||||
import kr.co.vividnext.sodalive.common.SharedPreferenceManager
|
||||
|
||||
class NewCharactersAllViewModel(
|
||||
private val repository: NewCharactersRepository
|
||||
) : BaseViewModel() {
|
||||
|
||||
private val _toastLiveData = MutableLiveData<String?>()
|
||||
val toastLiveData: LiveData<String?> get() = _toastLiveData
|
||||
|
||||
private val _isLoading = MutableLiveData(false)
|
||||
val isLoading: LiveData<Boolean> get() = _isLoading
|
||||
|
||||
private val _totalCount = MutableLiveData<Long>(0)
|
||||
val totalCount: LiveData<Long> get() = _totalCount
|
||||
|
||||
private val _items = MutableLiveData<List<Character>>(emptyList())
|
||||
val items: LiveData<List<Character>> get() = _items
|
||||
|
||||
private var page = 0
|
||||
private val size = 20
|
||||
private var isLast = false
|
||||
|
||||
fun loadMore() {
|
||||
if (_isLoading.value == true || isLast) return
|
||||
_isLoading.value = true
|
||||
|
||||
compositeDisposable.add(
|
||||
repository.getRecentCharacters(
|
||||
token = "Bearer ${SharedPreferenceManager.token}",
|
||||
page = page,
|
||||
size = size
|
||||
)
|
||||
.subscribeOn(Schedulers.io())
|
||||
.observeOn(AndroidSchedulers.mainThread())
|
||||
.subscribe({ response ->
|
||||
val data = response.data
|
||||
if (response.success && data != null) {
|
||||
val current = _items.value ?: emptyList()
|
||||
val next = current + data.content
|
||||
_items.value = next
|
||||
_totalCount.value = data.totalCount
|
||||
if (data.content.isNotEmpty()) {
|
||||
page += 1
|
||||
} else {
|
||||
isLast = true
|
||||
}
|
||||
} else {
|
||||
_toastLiveData.value = response.message
|
||||
?: "알 수 없는 오류가 발생했습니다. 다시 시도해 주세요."
|
||||
}
|
||||
_isLoading.value = false
|
||||
}, { e ->
|
||||
_isLoading.value = false
|
||||
e.message?.let { Logger.e(it) }
|
||||
_toastLiveData.value = "알 수 없는 오류가 발생했습니다. 다시 시도해 주세요."
|
||||
})
|
||||
)
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,21 @@
|
||||
package kr.co.vividnext.sodalive.chat.character.newcharacters
|
||||
|
||||
import io.reactivex.rxjava3.core.Single
|
||||
import kr.co.vividnext.sodalive.chat.character.CharacterApi
|
||||
import kr.co.vividnext.sodalive.common.ApiResponse
|
||||
|
||||
class NewCharactersRepository(
|
||||
private val api: CharacterApi
|
||||
) {
|
||||
fun getRecentCharacters(
|
||||
token: String,
|
||||
page: Int,
|
||||
size: Int
|
||||
): Single<ApiResponse<RecentCharactersResponse>> {
|
||||
return api.getRecentCharacters(
|
||||
authHeader = token,
|
||||
page = page,
|
||||
size = size
|
||||
)
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,11 @@
|
||||
package kr.co.vividnext.sodalive.chat.character.newcharacters
|
||||
|
||||
import androidx.annotation.Keep
|
||||
import com.google.gson.annotations.SerializedName
|
||||
import kr.co.vividnext.sodalive.chat.character.Character
|
||||
|
||||
@Keep
|
||||
data class RecentCharactersResponse(
|
||||
@SerializedName("totalCount") val totalCount: Long,
|
||||
@SerializedName("content") val content: List<Character>
|
||||
)
|
||||
Reference in New Issue
Block a user