دوره زبان تخصصی برای برنامه‌نویسان (هدیه ویژه ثبت‌نام در دوره‌های متخصص) (فرصت محدود ⏰)
۰ ثانیه
۰ دقیقه
۰ ساعت
۱ علی خدری
viewModel and duplicate data
پوریا شفیعی حل شده توسط پوریا شفیعی

سلام 

من یک پروژه خبری بر اساس معماری MVVM و زبان کاتلین و RxJava , liveData ساختم.

وقتی لیست خبرها رو می‌گیرم و وارد جزئیات خبر می‌شوم و دوباره به فرگمنت لیست خبر برمیگردم  دو تا اتفاق می‌افتد.

اول اینکه لیست خبرهایی قبلا وجود داشته نمایش می‌دهد.

دوم اینکه باز به سمت سرور رکوئست می‌فرستد و دوباره لیست خبرها رو آپدیت می‌کند.

خب من نمی‌خوام وقتی برمیگردم به صفحه لیست خبرها دوباره لیست خبرها آپدیت شود.

چه راه حلی پیشنهاد می‌دهید؟

class NewsViewModel:ViewModel() {
    private val compositeDisposable=CompositeDisposable()
    private var news= MutableLiveData<NewsLab>()
    private val _str= MutableLiveData<String>()
    val str:LiveData<String>
        get() =_str
         fun getNews(context: Context):LiveData<NewsLab>{
        LabRetrofit.getInstance(context).news()
            .subscribeOn(Schedulers.newThread())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(object : SingleObserver<NewsLab> {
                override fun onSubscribe(d: Disposable) {
                    compositeDisposable.add(d)
                }
                override fun onSuccess(t: NewsLab) {
                  if (t.success){
                      news.value=t
                      Log.i("newsViewModel", t.dataNews!!.get(0).toString())
                  }else{
                      _str.value="خطا"
                  }
                }
                override fun onError(e: Throwable) {
                    _str.value=e.message
                }
            })
        return news
    }
    override fun onCleared() {
        super.onCleared()
        compositeDisposable.clear()
    }
}

 

 

 

سلام و احترام

 

 

قبل از هر چیز پیشنهاد میکنم جلسه وبینار رو مشاهده کنید این موضوع با زبان جاوا (rx,livedata) تدریس شده .اما در زیر راه حل دیگری با کاتلین برای شما قرار میدم.

سناریویی که من در پروژه‌ها انجام میدم بدین شکل هست(ترکیب کوروتین و رتروفیت و لایو دیتا) که خدمتتون عرض میکنم:

 

۱- نوشتن متد مربوطه (مثلا گرفتن لیست خبر ها)در Repository 

صرفا یک مثال  :(البته من از روش Paging و Hilt استفاده کردم) ولی خب فرقی در اصل مسءله نمیکند.

 

@Singleton
class TaskRepository @Inject constructor(private val taskApi: ITaskApi) {
    fun getSearchResults(query: String)=
        Pager(
            config = PagingConfig(
                pageSize = 20,
                maxSize = 100,
                enablePlaceholders = false
            ),
            pagingSourceFactory = { TaskPagingSource(UTaskApi, query) }
        ).liveData
}

 

 

۲- سپس در ViewModel پیاده سازی موارد مربوطه به صورت livedata (البته من ویو مدلی نوشتم که به صورت پیشفرض سرچ تسک‌های پیشفرض رو انجام بده که شما اگر میخواهید از این روش استفاده کنید باید بجای سرچ از گرفتن لیست خبر‌ها این موضوع رو هندل کنید چرا که این مورد صرفا یک مثال هست) :

class TaskViewModel @ViewModelInject constructor(private val repository: TaskRepository) :
    ViewModel() {
    private val currentQuery = MutableLiveData(Default_QUERY_FOR_SEARCH)
    val tasks =currentQuery.switchMap { queryString->
        repository.getSearchResults(queryString).cachedIn(viewModelScope)
    }
    fun searchTasks(query:String){
        currentQuery.value = query
    }
    companion object{
        private const val Default_QUERY_FOR_SEARCH = "Test"
    }
}

۳- سپس پیاده سازی موارد در ویو مدل و observe کردنشان در Fragment:

//---

۴-پیاده سازی کلیک کردن کاربر روی ایتم مربوطه(برای مثال شما خبر مربوطه در adapter):

 

init {
            binding.root.setOnClickListener {
                val position = bindingAdapterPosition
                if (position != RecyclerView.NO_POSITION) {
                    val item = getItem(position)
                    if (item != null) {
                        listener.onItemClick(item)
                    }
                }
            }
        }

 

۵- implements کردن اینترفیسی که در adapter ساختم در Fragment مربوطه که دیتا رو به DetailFragment منتقل کردم :

override fun onItemClick(task: Task) {
        val action = TaskFragmentDirections.actionTaskFragmentToDetailsFragment(task)
        findNavController().navigate(action)
    }

با این روش مشکل شما حل خواهید شد; یعنی وقتی در فرگمنت جزییات back میزنید دیتا از اول از سرور دریافت نمیشود.

اگر هم میخواهید از روش rx استفاده کنید پیشنهاد من به شما دیدن وبینار مربوطه هست که این موضوع تدریس شده.

امیدوارم سر نخ خوبی بهتون داده باشم برای پیاده سازی پروژه تون

سالم و موفق باشید

 

 

 

 

 

 

 

بهترین پاسخ
پوریا شفیعی ۱۱ آذر ۱۳۹۹، ۰۶:۴۶