我正在使用MVVM体系结构来构建简单的订购应用程序。我在我的ProductsFragment中使用RecyclerView列出了所有可以订购的产品。我还在ViewModel中使用LiveData,在Fragment中可以观察到它,以检查对产品列表的任何更改。
在我的列表项中,我有3个按钮:1个按钮,用于将产品添加到购物篮中,另一个按钮,用于增加客户想要添加到购物篮中的数量,第三个按钮用于减少客户想要添加到购物篮中的数量。
在我的产品数据类中,每当客户单击增量或减量按钮时,数量都会在数据模型中更新。
我还使用数据绑定将产品绑定到recyclerview列表项布局和Click侦听器。我正在使用ListAdapter开箱即用地访问DiffUtil。
我遇到的问题是,当通知了observable时,我想使用ListAdapter的SubmitList方法,以便在RecyclerView中仅更新已更改的项目。但是,我注意到DiffUtil方法areContentsTheSame()始终返回true。因此,列表项未更新。我不想使用notifyDatasetChanged,因为这会阻塞UI线程。
我遇到的另一个问题是,当我向购物篮中添加产品时,会保留对该产品的引用,因此当我向购物篮中添加产品时,当我只想更新MutableLiveData时,MutableLiveData也将更新。将产品添加到购物篮时,如何停止对正在创建的LiveData的引用?
产品浏览模型
class ProductsViewModel : ViewModel() { // LIVE DATA private val _basket = MutableLiveData() val basket: LiveData get() = _basket private val _products = MutableLiveData >() val products: LiveData
> get() = _products init { _basket.value = Basket() _products.value = dummyData } fun onIncrementProductQuantityButtonPressed(product: Product) { //product.quantity += 1 //val newList = _products.value //_products.value = newList val newProduct = product.copy(quantity = product.quantity.plus(1)) _basket.value!!.updateProductInBasket(newProduct) _basket.value = _basket.value } fun onDecrementProductQuantityButtonPressed(product: Product) { if (product.quantity>1) { //product.quantity = product.quantity.minus(1) val newProduct = product.copy(quantity = product.quantity.minus(1)) _basket.value!!.updateProductInBasket(newProduct) _basket.value = _basket.value } } }
产品片段
class ProductsFragment : Fragment() { private lateinit var viewModel: ProductsViewModel override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { // Inflate the layout for this fragment val binding: FragmentProductsBinding = DataBindingUtil.inflate( inflater, R.layout.fragment_products, container, false) viewModel = ViewModelProviders.of(this).get(ProductsViewModel::class.java) val adapter = ProductsAdapter(ProductListener { product, onClickType -> when(onClickType) { OnClickType.INCREMENT -> { viewModel.onIncrementProductQuantityButtonPressed(product) } OnClickType.DECREMENT -> { viewModel.onDecrementProductQuantityButtonPressed(product) } OnClickType.BASKET -> { viewModel.addToBasketButtonPressed(product) } } }) viewModel.products.observe(this, Observer { list -> adapter.submitList(list) //adapter.notifyDataSetChanged() // TODO: check why I have to do notifyDataSetChanged() }) viewModel.basket.observe(this, Observer { activity?.invalidateOptionsMenu() }) binding.viewModel = viewModel binding.lifecycleOwner = this binding.productsRecyclerView.adapter = adapter setHasOptionsMenu(true) return binding.root }
产品适配器
class ProductsAdapter(private val clickListener: ProductListener) : ListAdapter(ProductDiffUtil()) { class ProductDiffUtil: DiffUtil.ItemCallback () { override fun areItemsTheSame(oldItem: Product, newItem: Product): Boolean { Log.d("Products", "Are items the same") return oldItem.name == newItem.name && oldItem.size == newItem.size } override fun areContentsTheSame(oldItem: Product, newItem: Product): Boolean { Log.d("Products", "Are contents the same ${oldItem == newItem}") Timber.d("Are contents the same ${oldItem == newItem}") Timber.d("OLD ITEM: $oldItem") Timber.d("NEW ITEM: $newItem") return oldItem == newItem //need to check this } } override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ProductViewHolder { return ProductViewHolder.from(parent) } override fun onBindViewHolder(holder: ProductViewHolder, position: Int) { val item = getItem(position) //use holder to access all the views in the card item holder.bind(clickListener, item) } class ProductViewHolder private constructor(private val binding: LayoutProductCardBinding) : RecyclerView.ViewHolder(binding.root) { fun bind(clickListener: ProductListener, item: Product) { binding.product = item binding.clickListener = clickListener binding.executePendingBindings() } companion object { fun from(parent: ViewGroup): ProductViewHolder { val layoutInflater = LayoutInflater.from(parent.context) val binding = LayoutProductCardBinding.inflate(layoutInflater, parent, false) return ProductViewHolder(binding) } } } } class ProductListener(val clickListener: (product: Product, clickType: OnClickType) -> Unit) { fun onAddToBasket(product: Product) = clickListener(product, OnClickType.BASKET) fun onDecrementProductQuantity(product: Product) = clickListener(product, OnClickType.DECREMENT) fun onIncrementProductQuantity(product: Product) = clickListener(product, OnClickType.INCREMENT) } enum class OnClickType { BASKET, DECREMENT, INCREMENT }
小智.. 5
您的问题似乎有一个简单的解决方案:)。您正在做的是仅更新从视图传递到VM的产品的价值。您应该做的是:使用.copy(quantity = ...)方法创建产品的副本,覆盖数量。然后替换列表中的上一个项目,并将新列表传递给LiveData。您可能需要主动调用notifyDataSetChanged(根据代码中的注释)也可能是这种情况。
希望这可以帮助。干杯!
您的问题似乎有一个简单的解决方案:)。您正在做的是仅更新从视图传递到VM的产品的价值。您应该做的是:使用.copy(quantity = ...)方法创建产品的副本,覆盖数量。然后替换列表中的上一个项目,并将新列表传递给LiveData。您可能需要主动调用notifyDataSetChanged(根据代码中的注释)也可能是这种情况。
希望这可以帮助。干杯!