- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我是 Android 的新手,我正在尝试使用新架构组件实现条码阅读器场景。
每次读取条码时,如果条码不在列表中,我想更新 ViewModel 中的列表,添加新元素,否则增加数量。
以下解决方案有效,但我对在适配器上调用“notifyDataSetChanged”以更新 UI 并不满意。这是因为 ViewModel 列表和适配器内部列表包含对相同对象的引用,因此 DiffUtil 不会捕获任何更改。
是否有更好的更新 UI 的方法?除了适配器之外,我还应该考虑进行任何改进来处理架构组件吗?
View 模型
public class ScannerViewModel extends ViewModel {
private MutableLiveData<List<ProductScan>> scanListLD;
public ScannerViewModel() {
scanListLD = new MutableLiveData<>();
scanListLD.setValue(new ArrayList<ProductScan>());
}
public LiveData<List<ProductScan>> getScanList() {
return scanListLD;
}
public void addBarcode(String barcode) {
List<ProductScan> list = scanListLD.getValue();
ProductScan scan = null;
for (ProductScan item : list) {
if (item.barcode.equals(barcode)) {
scan = item;
break;
}
}
if (scan == null) {
scan = new ProductScan();
scan.barcode = barcode;
scan.qt = 0;
list.add(scan);
}
scan.qt += 1;
scanListLD.postValue(list);
}
}
适配器
public class ScannerAdapter extends RecyclerView.Adapter<ScannerAdapter.ViewHolder> {
private List<ProductScan> scans;
public interface OnItemClickListener {
void onItemClick();
}
public ScannerAdapter(List<ProductScan> scans) {
this.scans = scans;
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.template_scan, parent, false);
ScannerAdapter.ViewHolder viewHolder = new ScannerAdapter.ViewHolder(view);
return viewHolder;
}
@Override
public void onBindViewHolder(ViewHolder holder, int position) {
holder.bindData(scans.get(position), position);
}
@Override
public int getItemCount() {
return scans != null ? scans.size() : 0;
}
//Not used since scans and newScans contain same objects
public void setScans(final List<ProductScan> newScans) {
if (scans == null) {
scans = new ArrayList<ProductScan>();
scans.addAll(newScans);
notifyItemRangeInserted(0, newScans.size());
} else {
DiffUtil.DiffResult result = DiffUtil.calculateDiff(new DiffUtil.Callback() {
@Override
public int getOldListSize() {
return scans != null ? scans.size() : 0;
}
@Override
public int getNewListSize() {
return newScans != null ? newScans.size() : 0;
}
@Override
public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
ProductScan oldScan = scans.get(oldItemPosition);
ProductScan newScan = newScans.get(newItemPosition);
return Utils.equals(oldScan.barcode,newScan.barcode);
}
@Override
public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) {
ProductScan oldScan = scans.get(oldItemPosition);
ProductScan newScan = newScans.get(newItemPosition);
return Utils.equals(newScan.barcode, oldScan.barcode)
&& Utils.equals(newScan.productId, oldScan.productId)
&& Utils.equals(newScan.productDescription, oldScan.productDescription)
&& Utils.equals(newScan.qt, oldScan.qt);
}
});
scans.clear();
scans.addAll(newScans);
result.dispatchUpdatesTo(this);
}
}
public class ViewHolder extends RecyclerView.ViewHolder {
private TemplateScanBinding binding;
public ViewHolder(View itemView) {
super(itemView);
binding = DataBindingUtil.bind(itemView);
}
public void bindData(ProductScan scan, int position) {
binding.setScan(scan);
binding.setBtnIncreaseQtListener(() -> {
scan.qt += 1;
notifyItemChanged(position);
});
binding.setBtnDecreaseQtListener(() -> {
scan.qt = Math.max(1, scan.qt - 1);
notifyItemChanged(position);
});
binding.edtQt.setOnEditorActionListener(new TextView.OnEditorActionListener() {
@Override
public boolean onEditorAction(TextView textView, int i, KeyEvent keyEvent) {
if (i == EditorInfo.IME_ACTION_DONE) {
binding.edtQt.clearFocus();
}
return false;
}
});
binding.edtQt.setOnFocusChangeListener(new View.OnFocusChangeListener() {
@Override
public void onFocusChange(View view, boolean b) {
if (scan.qt == null || scan.qt < 1) {
scan.qt = 1;
notifyItemChanged(position);
}
InputMethodManager imm = (InputMethodManager) binding.getRoot().getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
imm.hideSoftInputFromWindow(view.getWindowToken(), 0);
}
});
}
}
}
Activity
public class ScannerActivity extends ScannerBaseActivity {
@Inject
ViewModelFactory viewModelFactory;
private ScannerViewModel viewModel;
private ActivityScannerBinding binding;
private ScannerAdapter adapter;
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
((MyApplication) getApplication()).getComponent().inject(this);
viewModel = ViewModelProviders.of(this, viewModelFactory).get(ScannerViewModel.class);
binding = DataBindingUtil.setContentView(this, R.layout.activity_scanner);
adapter = new ScannerAdapter(viewModel.getScanList().getValue());
binding.recvScans.setLayoutManager(new LinearLayoutManager(this));
binding.recvScans.setAdapter(adapter);
viewModel.getScanList().observe(this, (list) -> {
adapter.notifyDataSetChanged();
});
binding.btnScan.setOnClickListener((v) -> {
//calls viewModel.addBarcode(String barcode)
readBarcode(readBarcodeCallback);
});
}
}
最佳答案
您可以使用 PagedListAdapterHelper在您现有的适配器中,如下所示
class UserAdapter extends RecyclerView.Adapter<UserViewHolder> {
private final PagedListAdapterHelper<User> mHelper;
public UserAdapter(PagedListAdapterHelper.Builder<User> builder) {
mHelper = new PagedListAdapterHelper(this, DIFF_CALLBACK);
}
@Override
public int getItemCount() {
return mHelper.getItemCount();
}
public void setList(PagedList<User> pagedList) {
mHelper.setList(pagedList);
}
@Override
public void onBindViewHolder(UserViewHolder holder, int position) {
User user = mHelper.getItem(position);
if (user != null) {
holder.bindTo(user);
} else {
// Null defines a placeholder item - PagedListAdapterHelper will automatically
// invalidate this row when the actual object is loaded from the database
holder.clear();
}
}
public static final DiffCallback<User> DIFF_CALLBACK = new DiffCallback<User>() {
@Override
public boolean areItemsTheSame(
@NonNull User oldUser, @NonNull User newUser) {
// User properties may have changed if reloaded from the DB, but ID is fixed
return oldUser.getId() == newUser.getId();
}
@Override
public boolean areContentsTheSame(
@NonNull User oldUser, @NonNull User newUser) {
// NOTE: if you use equals, your object must properly override Object#equals()
// Incorrectly returning false here will result in too many animations.
return oldUser.equals(newUser);
}
}
或
class UserAdapter extends PagedListAdapter<User, UserViewHolder> {
public UserAdapter() {
super(DIFF_CALLBACK);
}
@Override
public void onBindViewHolder(UserViewHolder holder, int position) {
User user = getItem(position);
if (user != null) {
holder.bindTo(user);
} else {
// Null defines a placeholder item - PagedListAdapter will automatically invalidate
// this row when the actual object is loaded from the database
holder.clear();
}
}
public static final DiffCallback<User> DIFF_CALLBACK = new DiffCallback<User>() {
@Override
public boolean areItemsTheSame(
@NonNull User oldUser, @NonNull User newUser) {
// User properties may have changed if reloaded from the DB, but ID is fixed
return oldUser.getId() == newUser.getId();
}
@Override
public boolean areContentsTheSame(
@NonNull User oldUser, @NonNull User newUser) {
// NOTE: if you use equals, your object must properly override Object#equals()
// Incorrectly returning false here will result in too many animations.
return oldUser.equals(newUser);
}
}
关于android - 实时数据列表 - 适配器不使用 DiffUtil 方法更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46771925/
我们正在尝试对包含来自 sqlite 数据库的数据的 ArrayList 进行排序。 我可以通过添加到查询“ORDER BY”+ Col_IDBG+“DESC 来对列表进行排序 除了 Recycler
我通常在需要时将 diffutil 与 Recyclerview 一起使用。现在我有这样的情况,我从后端得到的项目是这样的: data class CarouselItem(var url: Stri
我正在尝试使用DiffUtil来更新RecyclerView。我有一个类 DataProvider ,它是一个单例类,它保存包含对象的列表。 每次我需要更新RecyclerView时,我都会修改Dat
我有一个 RecyclerView,其中的元素通过 WebSocket 更新。此更新可能会影响列表中不同位置的多个元素。我想用 DiffUtil更新 RecycleView 的元素。套接字更新本身不包
DiffUtils.dispatchUpdatesTo(adapter) 没有更新我在 recyclerView 中的数组大小 我的 result 的大小大于当前的 itemList,但只显示当前的大
我为 recyclerView 制作了一个适配器,并使用 DiffUtil 以更漂亮的方式显示列表的更新。我使用 google codelabs 作为引用。但该列表并未从以下代码更新。请帮忙 clas
我正在使用 diff util 来提高我的 recyclerview 的性能,而不是调用 notifyDataSetChanged()。 recyclerview 有一个带有一些芯片的标题,可以通过
我有一个具有 DiffUtil 功能的 RecyclerView 适配器。 我想在 DiffUtil 完成其魔法时使用 DataObserver 通知 fragment 。但是,看起来 DiffUti
我的回收站 View 中有无限滚动,因此,它会在有新数据时更新。我正在使用 DiffUtil 更新 recyclerview 中的数据。 DiffUtil 确实会更新数据,但是只要有更新数据,recy
一段时间以来,我一直被困在一个相当简单的问题上。出于某种原因,我无法在 StackOverflow 上找到这方面的答案,所以我想我应该直接寻求帮助。 问题很简单,我有一个按日期排序的交易对象列表。我正
我正在使用下面的代码来测试 DiffUtil实现。根据文档- DiffUtil is a utility class that can calculate the difference between
更新:其中一个问题解决了:现在updateList解决了,问题是我把mAdapter定义为RecyclerView .Adapter 而不是 MyAdapter。但是现在即使我正在获取数据,列表中也没
目前我尝试在我的项目中使用DiffUtil来通知RecyclerView的项目。 与此同时,我编写了一个可以添加页眉和页脚的自定义适配器。我使用了这个适配器并添加了一个加载更多的页脚,这样我的列表在向
我对 Android 还很陌生,目前正在开发我的第一个基于 RecyclerView 的严肃应用程序。当我插入或更新现有行时,我注意到我的应用程序中有一些有问题的行为:即使我在添加或 notifyIt
我使用 ListAdapter 的绑定(bind)和 DiffUtil.ItemCallback 的定义。删除项目(至少 2 个)时出现 IndexOutOfBoundsException。列表的更新
每次我调用无效数据时,我的 DIFF_UTIL 都不会被使用。日志未显示,整个列表已更新为新数据,导致屏幕移动位置等。不确定这里的问题是什么。 我有PagedListAdapter与 LiveData
我无法将 Kotlin Flows 和异步 DiffUtil 放在一起。 我的 RecyclerView.Adapter 中有这个函数,它在计算线程上计算 DiffUtil 并将更新发送到主线程上的
我是 Android 的新手,我正在尝试使用新架构组件实现条码阅读器场景。 每次读取条码时,如果条码不在列表中,我想更新 ViewModel 中的列表,添加新元素,否则增加数量。 以下解决方案有效,但
我遇到了一个有趣的情况,我的适配器有 2 种 View 类型 - HEADER 和 ITEM。 header View 始终位于位置 0。更新数据时,我使用 DiffUtil 获取数据的差异。为了分派
我的 diffcallback areContentsTheSame(oldItem: ItemModel, newItem: ItemModel) 总是接收相同的内容。我用状态来检查,但每次状态都是
我是一名优秀的程序员,十分优秀!