从DialogFragment和主Activity更新RecyclerView

3

什么是最好的模式来实现它。我做了一些解决方法,但我感觉这是一个不好的模式。

我的情况是这样的。

我有一个主Activity。

MainActivity.java

在这个Activity中,我加载带有RecyclerView的Fragment。

ListFragment.java

我还有DialogFragmentItemAdapter

从适配器中,我显示DialogFragment。

完整的适配器代码

public class TypeAdapter extends RecyclerView.Adapter<CostTypeAdapter.ViewHolder> {
    private static final String TAG = "TypeAdapter";
    private List<CostType> mCostsType;
    private Context mContext;

public CostTypeAdapter(Context context, List<CostType> coststype) {
    mContext = context;
    mCostsType = coststype;

}

@Override
public ViewHolder onCreateViewHolder(ViewGroup viewGroup, int viewType) {
    // Create a new view.
    View v = LayoutInflater.from(viewGroup.getContext())
            .inflate(R.layout.costtype_row, viewGroup, false);

    return new ViewHolder(v);
}

// BEGIN_INCLUDE(recyclerViewOnBindViewHolder)
@Override
public void onBindViewHolder(final ViewHolder viewHolder, final int position) {
    //Log.d(TAG, "Element " + position + " set.");
    String Name = mCostsType.get(position).getName();
    viewHolder.tvName.setText(Name);

    //mView click
    viewHolder.cardView.setPreventCornerOverlap(false);
    viewHolder.cardView.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            //Dialog here
            FragmentActivity activity = (FragmentActivity)(mContext);
            FragmentManager fm = activity.getSupportFragmentManager();

            CostType costType = new CostType();
            costType.setTypeID(mCostsType.get(position).getTypeID());
            costType.setName(mType.get(position).getName());
            costType.setPriority(0);
            CostsTypeDialogFragment alertDialog = CostsTypeDialogFragment.newInstance(costType);
            alertDialog.show(fm, "cost_type");

        }
    });


}
// END_INCLUDE(recyclerViewOnCreateViewHolder)

public void refreshList() {
    this.notifyDataSetChanged();
    System.out.println("refreshList");

}



@Override
public int getItemCount() {
    return mType.size();
}

/**
 * Provide a reference to the type of views that you are using (custom ViewHolder)
 */
public static class ViewHolder extends RecyclerView.ViewHolder {
    private final TextView tvName;
    private final CardView cardView;

    public ViewHolder(View v) {
        super(v);

        v.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Log.d(TAG, "Element " + getPosition() + " clicked.");

            }
        });
        tvName = (TextView) v.findViewById(R.id.tvName);
        cardView = (CardView) v.findViewById(R.id.card_view);
    }

}



}

DialogFragment.java 我有一个监听器用于与MainActivity通信的对话框

    onDialogCloseListener mListener;
    // Container Activity must implement this interface
    public interface onDialogCloseListener {
        public void onDialogClose();
    }

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        try {
            mListener = (onDialogCloseListener) activity;
        } catch (ClassCastException e) {
            throw new ClassCastException(activity.toString() + " must implement OnDialogClose listener");
        }
    }
    private List<MyType> mType;

    ...

从对话框中的 onClick:

mListener.onDialogClose();

这个对话框有一些按钮。其中一个例如是修改/删除回收站列表上的项目。在这个对话框中,我有一个onDialogCloseListener接口,在我的MainActivity中使用它。

MainActivity.java

public class MainActivity extends BaseActivity implements DialogFragment.onDialogCloseListener {

    ... 
    @Override
    public void onDialogClose() {

        RecyclerView mRecyclerView;
        mRecyclerView = (RecyclerView) findViewById(R.id.recyclerView);

        //not working
        mRecyclerView.getAdapter().notifyDataSetChanged();
    }

我可以这样从Activity中使用mRecyclerView。例如,我可以使其不可见。但是notifyDataSetChanged无法正常工作。
我的解决方法是创建新的适配器-新的sql查询、新的对象列表并将其再次设置为RecyclerView。但这可能不是一个好方法。因为我无法保持滚动位置。
但如何与适配器建立连接?我希望能够添加动画,例如删除项目。
编辑:运行
mRecyclerView.getAdapter().notifyItemRemoved(0);

在onDialogClose中很好地工作。列表已刷新。但是当我想用相同的方式来处理

时,出现了问题。
mRecyclerView.getAdapter().notifyDataSetChanged();

或者
mRecyclerView.getAdapter().notifyItemChanged(0);

列表没有刷新。

你只需要在适配器上通知数据集已更改吗? - Carlo Moretti
@Onheiron,这可能是最简单的方法,但我读到RecyclerView并不那么容易。如果只使用notifyDataSetChanged选项,那么是的。但我需要能够从适配器中获取项目。例如,在删除时显示动画,保持滚动位置。 - adek
你能发布你完整的适配器代码吗? - Carlo Moretti
@Onheiron 确定。已编辑。 - adek
另外一件事。这个是有效的: mRecyclerView.getAdapter().notifyItemRemoved(0)删除很好用。我可以看到动画。所以问题是为了编辑记录实现类似的效果。例如: mRecyclerView.getAdapter().notifyItemChanged(16)不会刷新列表。 - adek
1个回答

5

建议 1:不要在适配器类中显示对话框,使您的onClick触发您自己在Activity中的监听器,例如onItemClicked(CostType costType)

class MyAdapter extends RecyclerView.Adapter .. {

    public interface OnItemClickListener {
        void onItemClicked(CostType costType);
    }

    private OnItemClickListener mListener;

    // setter and getter for OnitemClickListener

    // inside onBindViewHolder
    viewHolder.cardView.setOnClickListener(new View.OnClickListener() {

      @Override
      public void onClick(View v) {
        CostType costType = new CostType();
        costType.setTypeID(mCostsType.get(position).getTypeID());
        costType.setName(mType.get(position).getName());
        costType.setPriority(0);
        mListener.onItemClicked(costType);
      }

}

class MyActivity extends Activity {

    // inside onCreate
    adapter.setOnItemClickListener(new ...);

}

建议2:如果您需要将Fragment B的结果传递给Fragment A,您不需要使用activity,而是使用setTargetFragment / getTargetFragment方法。

// inside MyListFragment
MyDialogFragment frag = MyDialogFragment.newInstance();
frag.setTargetFragment(this, 0);
// commit transaction

// inside MyDialogFragment to trigger listener, make sure MyDialogFragment
// implement OnDialogCloseListener interface
((MyListFragment)getTargetFragment()).onDialogClose();

谢谢。这就是我为什么要寻求好的模式的原因。所以我应该将我的onClick移动到MainActivity(而不是片段活动)。但是如何做到这一点呢?我可以像DialogFragment监听器一样做吗? - adek
@adek 尝试做以下事情: 1)确保 onDialogClose 方法被调用。 2)在调用 notifyDataSetChanged 方法后,是否会调用 onBindViewHolder 方法? 3)在创建适配器时,请将适配器作为活动的类变量保留引用。例如,将 mRecyclerView.getAdapter().notifyDataSetChanged() 替换为 mAdapter.notifyDataSetChanged() - Dmytro Danylyk
好的,但mAdapter在FragmentList中。OnDialog called在MyActivity类中。OnAttach连接到主activity。但是从主activity中,我能够获取mRecyclerView参考 - 例如mRecyclerView可见性改变有效。所以在我看来getAdapter()应该可以工作。 - adek
@adek 你是对的,你的代码应该可以正常工作。我已经添加了另一个建议2。 - Dmytro Danylyk
让我们在聊天中继续这个讨论 - Dmytro Danylyk
显示剩余2条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接