Как я могу проверить переработчик просмотра адаптера TextInputEditText из фрагмента?

У меня есть фрагмент, который содержит обзор переработчика. Управление этим обзором осуществляется с помощью адаптера, который включает в себя шаблон держателя. Во фрагменте у меня есть кнопка "Сохранить", эта кнопка должна проверить, что в списке нет пустых TextInputEditText.

Проблема в том, что я не могу получить доступ к TextInputEditText адаптера. ViewHolder доступен только из метода "onBindViewHolder". Я пытался получить доступ из фрагмента таким образом:

for (int i = 0; i < employeeList.size(); i++) {
            View employeeView = recyclerEmployees.findViewHolderForAdapterPosition(i).itemView;
            if (employeeView != null) {
                TextInputLayout textInputLayoutName = employeeView.findViewById(R.id.tilName);
                TextInputEditText textInputEditTextName = employeeView.findViewById(R.id.tieName);

Сделав это, мне удалось пометить ошибку в TextInputLayout. Я также был в состоянии получить текст TextInputEditText. Казалось бы, все работает, но...

Проблема появляется, когда список меняется, и мне нужно обновить адаптер, позвонив notifyDataSetChanged

findViewHolderForAdapterPosition возвращает ноль в некоторых позициях списка.

Я прочитал в каком-то посте, что этот метод не безопасно вызывать /questions/38568345/android-recyclerview-findviewholderforadapterposition-vozvraschaet-nol/38568361#38568361

Я также прочитал, что это не очень хорошая практика для доступа к viewholder вне onBindViewHolder метод.

Может ли кто-нибудь помочь мне получить это? Это не кажется сложным, но это доставляет мне большую потерю времени.

Я просто хочу нажать на кнопку Сохранить во фрагменте и проверить, что нет TextInputEditText пуст в списке предметов в моем recyclerview

Спасибо

3 ответа

Решение

Попробуй это

AddMoreAdapter

public class AddMoreAdapter extends RecyclerView.Adapter<AddMoreAdapter.ViewHolder> {
    Context context;
    ArrayList<AddMorePojo> arrayList;

    public AddMoreAdapter(Context context, ArrayList<AddMorePojo> arrayList) {
        this.context = context;
        this.arrayList = arrayList;
    }

    @Override
    public AddMoreAdapter.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {

        View view = LayoutInflater.from(context).inflate(R.layout.addmorelayout, parent, false);
        return new ViewHolder(view);
    }

    @Override
    public void onBindViewHolder(AddMoreAdapter.ViewHolder holder, int position) {


        Log.e("VALUE ", arrayList.get(position).getUserName());

        if (!TextUtils.isEmpty(arrayList.get(position).getUserName())) {

            holder.edtName.setText(arrayList.get(position).getUserName());
        }

        if (!TextUtils.isEmpty(arrayList.get(position).getError())) {
            holder.edtName.setError(arrayList.get(position).getError());
        } else {

            holder.edtName.setError(null);
        }


    }

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

    public ArrayList<AddMorePojo> getArrayList() {
        return arrayList;
    }

    public class ViewHolder extends RecyclerView.ViewHolder {

        EditText edtName;

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

            edtName = itemView.findViewById(R.id.edtUname);


            edtName.addTextChangedListener(new TextWatcher() {
                @Override
                public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {

                }

                @Override
                public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {

                    arrayList.get(getAdapterPosition()).setUserName(charSequence.toString());


                }

                @Override
                public void afterTextChanged(Editable editable) {

                }
            });


        }
    }
}

AddMorePojo

public class AddMorePojo
{
    String userName,error;

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    public String getError() {
        return error;
    }

    public void setError(String error) {
        this.error = error;
    }
}

MyActivity

public class MyActivity extends AppCompatActivity {


    RecyclerView myRc;
    ArrayList<AddMorePojo> arrayList = new ArrayList<>();
    Button btnGetData;
    AddMoreAdapter adapter;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_my);

        myRc = (RecyclerView) findViewById(R.id.myRc);
        btnGetData = (Button) findViewById(R.id.btnGetData);

        myRc.setHasFixedSize(true);
        myRc.setLayoutManager(new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false));

        AddMorePojo addMorePojo = new AddMorePojo();
        addMorePojo.setUserName("");
        arrayList.add(addMorePojo);

        AddMorePojo addMorePojo1 = new AddMorePojo();
        addMorePojo1.setUserName("");
        addMorePojo1.setError("");
        arrayList.add(addMorePojo1);


        AddMorePojo addMorePojo2 = new AddMorePojo();
        addMorePojo2.setUserName("");
        addMorePojo2.setError("");
        arrayList.add(addMorePojo2);



        adapter = new AddMoreAdapter(this, arrayList);
        myRc.setAdapter(adapter);

        btnGetData.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                boolean flag=true;
                ArrayList<AddMorePojo> pojoArrayList = adapter.getArrayList();

                for (int i = 0; i < pojoArrayList.size(); i++) {

                    if (pojoArrayList.get(i).getUserName().isEmpty()) {
                        pojoArrayList.get(i).setError("Please enter userName");
                        flag=false;
                    }else {
                     //   pojoArrayList.get(i).setError("");
                    }
                }
                adapter = new AddMoreAdapter(MyActivity.this, pojoArrayList);
                myRc.setAdapter(adapter);

                if (flag){
                    Toast.makeText(MyActivity.this, "Valid data", Toast.LENGTH_SHORT).show();
                }else {
                    Toast.makeText(MyActivity.this, "Invalid data", Toast.LENGTH_SHORT).show();
                }
            }

        });
    }


}

layout.activity_my

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="horizontal">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/myRc"
        android:layout_width="match_parent"
        android:paddingBottom="50dp"
        android:layout_height="match_parent" />


    <Button
        android:id="@+id/btnGetData"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:padding="20dp"
        android:gravity="center"
        android:layout_alignParentBottom="true"
        android:text="Get Data" />
</RelativeLayout>

addmorelayout

<?xml version="1.0" encoding="utf-8"?>
<android.support.v7.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    app:cardElevation="10dp"
    app:cardUseCompatPadding="true">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical">

        <EditText
            android:id="@+id/edtUname"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="Enter User Name" />


    </LinearLayout>

</android.support.v7.widget.CardView>

Да, вы правы, это не сложно.

@Override
public void onBindView(Object object, int pos) {

        Employee employee = (Employee) object;
        editText.addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {

            }

            @Override
            public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {
                employee.setName(charSequence.toString());
            }

            @Override
            public void afterTextChanged(Editable editable) {

            }
        });
    }

А при нажатии кнопки сохранить значение из списка.

Например, вы можете использовать логическое поле в вашей модели.

class TestData{

 private boolean isTextEmpty;

 private String textData;

 public void setTextEmpty(boolean isTrue){
    isTextEmpty = isTrue;
 }

 public boolean isTextEmpty(){
    return isTextEmpty;
 }

 public void setTextData(String data){
    textData = data;
 }

 public void getData(){
    return textData;
 }

 ........
 ///other fields goes here
}

теперь используйте эту модель в своем списке и скормите ее адаптеру

    public class TestAdapter extends RecyclerView.Adapter<TestAdapter.ViewHolder> {

        ArrayList<TestData> arrayList;
        boolean isSubmitClick = false;

        public AddMoreAdapter(ArrayList<TestData> arrayList) {
            this.arrayList = arrayList;
        }

        @Override
        public TestAdapter.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {

            View view = LayoutInflater.from(context).inflate(R.layout.test_data_item, parent, false);
            return new ViewHolder(view);
        }

        @Override
        public void onBindViewHolder(TestAdapter.ViewHolder holder, int position) {
            TestData data = arrayList.get(holder.getAdapterPosition());
            if (holder.editText.getText().toString()==null) {
                if(isSubmitClick){
                 holder.inputText.setError("please enter a valid input");
                }   
                data.setTextEmpty(true);
            } else {
                data.setTextEmpty(false);
                //do something else
            }
        }

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

        public class ViewHolder extends RecyclerView.ViewHolder {
            TextInputLayout textInput;
            EditText editName;

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

                editName = itemView.findViewById(R.id.etTest);
                textInput = itemView.findViewById(R.id.tilTest);

                editName.addTextChangedListener(new TextWatcher() {
                    @Override
                    public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {

                    }

                    @Override
                    public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {

                    }

                    @Override
                    public void afterTextChanged(Editable editable) {
                         arrayList.get(getAdapterPosition()).setTestData(editable.toString());
                    }
                });

            }
        }

        public void setSubmitClick(boolean isTrue){
          this.isSubmitClick = isTrue; 
        }

}

Теперь в вашем фрагменте сделать как этот вызов adapter.setSubmitClick(true); по нажатию вашей кнопки, а затем подтвердите данные, как показано ниже.

public void isDataValid(){

  for(int i = 0; i<arrayList.size(); i++){
     TestData data = arrayList.get(i);
     if(data.isTextEmpty()){
        adapter.notifyItemChanged(i);
        return;
     }
  }

  //do something here as all data is valid
  callServer();
}