Search code examples
androidlistandroid-listviewandroid-edittextandroid-arrayadapter

Android - Get each EditTexts values of ListView with custom ArrayAdapter


I've got a ListView with a customer ArrayAdapter and a custom ListItem-XML. At the bottom of this ListView I've added a FooterView with a Button. Now I want to get all the values of the three EditTexts in each of my ListItems when I click this Button.

Here below is my code:

activity_checklistresult.xml:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <ListView
        android:id="@android:id/list"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</RelativeLayout>

result_inner_view.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/item_layout"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="horizontal"
    android:gravity="center">

    <TextView
        android:id="@+id/tv_result_amount"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_margin="@dimen/default_margin" />
    <EditText
        android:id="@+id/et_result_amount"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_margin="@dimen/default_margin"
        android:inputType="number" />

    <LinearLayout
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_weight="1"
        android:gravity="center_vertical"
        android:orientation="vertical">

        <TextView
            android:id="@+id/tv_result_product_name"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:singleLine="true"
            android:ellipsize="end"
            android:layout_margin="@dimen/default_margin" />
        <AutoCompleteTextView
            android:id="@+id/actv_search_product"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:singleLine="true"
            android:ellipsize="end"
            android:layout_margin="@dimen/default_margin"
            android:inputType="textAutoComplete" />

    </LinearLayout>

    <TextView
        android:id="@+id/tv_result_price"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_margin="@dimen/default_margin" />
    <EditText
        android:id="@+id/et_result_price"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_margin="@dimen/default_margin"
        android:inputType="numberDecimal" />

</LinearLayout>

ResultListActivity.java:

public class ChecklistResultActivity extends ListActivity
{
    private MyResultAdapter adapt;
    private Button confirmButton;
    private ChecklistResultActivity rActivity;

    @Override
    public void onCreate(Bundle savedInstanceState){
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_checklistresult);

        // Set my Custom ArrayAdapter for this ListActivity
        setAdapter();

        rActivity = this;
    }

    private void setAdapter(){
        adapt = new MyResultAdapter(this, R.layout.result_inner_view, Controller.getInstance().getOrderedProducts());
        setListAdapter(adapt);

        // Add the confirm button at the end of the list
        addFooterConfirmButton();
    }

    private void addFooterConfirmButton(){
        ListView lv = (ListView)findViewById(android.R.id.list);
        FrameLayout footerLayout = (FrameLayout) getLayoutInflater().inflate(R.layout.result_footer_view, null);
        confirmButton = (Button)footerLayout.findViewById(R.id.confirm_result_button);
        confirmButton.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                ...

                // Get all EditText values (and compare them with the Default Value)
            }
        });
        lv.addFooterView(footerLayout);
    }

    ...

MyResultAdapter.java:

public class MyResultAdapter extends ArrayAdapter<OrderedProduct>
{
    private Context context;
    private int layoutResourceId;
    private LayoutInflater inflater;

    public MyResultAdapter(Context c, int layoutResourceId, List<OrderedProduct> objects){
        super(c, layoutResourceId, objects);
        this.layoutResourceId = layoutResourceId;
        this.context = c;
        inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent){
        View view = convertView;
        if(view == null)
            view = inflater.inflate(layoutResourceId, parent, false);

        ...

        return view;
    }
}

In the ResultListActivity at // Get all EditText values (and compare them with the Default Value) I want to loop through all the OrderedProducts and compare the default values of those, with the new filled-in values of the two EditTexts and the AutoCompleteTextView. So how can I get those values from within the onClick of the Footer-Button?

Thanks in advance for the responses.


Solution

  • Your adapter has to save text on change for each created EditText :

    1. create a map to save values
    2. Define and set a generic TextWatcher for every EditText
    3. set a Tag on the editText (using its position or any id you want) we'll use this tag as the key in the hashMap
    4. create a method to retrieve the value for a given EditText

    Adapter :

    public class MyResultAdapter extends ArrayAdapter<OrderedProduct>
    {
        ...
    
        private HashMap<String, String> textValues = new HashMap<String, String>();
    
        ...
    
        public ViewView getView(int position, View convertView, ViewGroup parent){
    
            View view = convertView;
            boolean convertViewWasNull = false;
            if(view == null){
                view = inflater.inflate(layoutResourceId, parent, false);
                convertViewWasNull = true;
            }
    
    
            myEditText1 = findViewById...
            myEditText2 = findViewById...
    
            if(convertViewWasNull ){
    
                //be aware that you shouldn't do this for each call on getView, just once by listItem when convertView is null
                myEditText1.addTextChangedListener(new GenericTextWatcher(myEditText1));
                myEditText2.addTextChangedListener(new GenericTextWatcher(myEditText2));
            }
    
            //whereas, this should be called on each getView call, to update view tags.
            myEditText1.setTag("theFirstEditTextAtPos:"+position);
            myEditText2.setTag("theSecondEditTextAtPos:"+position);
        }
    
    
        private class GenericTextWatcher implements TextWatcher{
    
           private View view;
           private GenericTextWatcher(View view) {
                this.view = view;
            }
    
            public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {}
            public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {}
    
            public void afterTextChanged(Editable editable) {
    
                String text = editable.toString();
                //save the value for the given tag :
                MyResultAdapter.this.textValues.put(view.getTag(), editable.toString());
            }
        }
    
        //you can implement a method like this one for each EditText with the list position as parameter :
        public String getValueFromFirstEditText(int position){
              //here you need to recreate the id for the first editText
             String result = textValues.get("theFirstEditTextAtPos:"+position);
             if(result ==null)
                  result = "default value";
    
             return result;
        }
    
        public String getValueFromSecondEditText(int position){
              //here you need to recreate the id for the second editText
             String result = textValues.get("theSecondEditTextAtPos:"+position);
             if(result ==null)
                  result = "default value";
    
             return result;
        }
    
    }