EditText теряет пользовательский ввод после прокрутки ListView

Мое приложение загружает данные из базы данных SQLite и загружает их в EditText внутри списка. Всякий раз, когда я изменяю значение редактируемого текста и прокручиваю список вниз, EditText возвращается к исходному значению, полученному из базы данных. Есть ли способ предотвратить это. Искал ответ в интернете, но так и не нашел. Любая помощь будет принята с благодарностью.

    public class ExerciseAdapter extends CursorAdapter {

    Context context;

    public ExerciseAdapter(Context context, Cursor c) {
        super(context, c, 0);
        this.context = context;
    }


    @Override
    public View newView(Context context, Cursor cursor, ViewGroup parent) {

        return LayoutInflater.from(context).inflate(R.layout.list_item_exercise, parent, false);
    }

    @Override
    public void bindView(View view, Context context, Cursor cursor) {
        int nameCol = cursor.getColumnIndex(WorkoutContract.WorkoutEntry.EXERCISE_NAME);
        String name = cursor.getString(nameCol);
        int weightCol = cursor.getColumnIndex(WorkoutContract.WorkoutEntry.WEIGHT);
        int weightInt = cursor.getInt(weightCol);
        String weight = String.valueOf(weightInt);
        int repCol = cursor.getColumnIndex(WorkoutContract.WorkoutEntry.REPS);
        int repsInt = cursor.getInt(repCol);
        String reps = String.valueOf(repsInt);
        int rpeCol = cursor.getColumnIndex(WorkoutContract.WorkoutEntry.RPE);
        int rpeInt = cursor.getInt(rpeCol);
        String rpe = String.valueOf(rpeInt);

        EditText exerciseName = (EditText) view.findViewById(R.id.exercise_name_table);
        exerciseName.setText(name);
        EditText exerciseWeight  = (EditText) view.findViewById(R.id.exercise_weight_table);
        exerciseWeight.setText(weight);
        EditText exerciseReps = (EditText) view.findViewById(R.id.exercise_reps_table);
        exerciseReps.setText(reps);
        EditText exerciseRpe = (EditText) view.findViewById(R.id.exercise_rpe_table);
        exerciseRpe.setText(rpe);

    }
}

    public class ExerciseActivity extends AppCompatActivity implements LoaderManager.LoaderCallbacks<Cursor> {

    private ExerciseAdapter adapter;
    Uri exerciseUri;

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


        ListView listView = findViewById(R.id.list);
        Intent intent = getIntent();
        exerciseUri = intent.getData();
        adapter = new ExerciseAdapter(this, null);
        listView.setAdapter(adapter);
        getLoaderManager().initLoader(0, null, this);
    }

    @Override
    public Loader<Cursor> onCreateLoader(int id, Bundle args) {
        String[] projections = {
                WorkoutContract.WorkoutEntry._ID,
                WorkoutContract.WorkoutEntry.DAY_ID,
                WorkoutContract.WorkoutEntry.EXERCISE_NAME,
                WorkoutContract.WorkoutEntry.WEIGHT,
                WorkoutContract.WorkoutEntry.REPS,
                WorkoutContract.WorkoutEntry.RPE
        };
        return new CursorLoader(
                this,
                exerciseUri,
                projections,
                null,
                null,
                null
        );
    }

    @Override
    public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
        adapter.swapCursor(data);
    }

    @Override
    public void onLoaderReset(Loader<Cursor> loader) {
        adapter.swapCursor(null);
    }
}

person Omar Carino    schedule 21.10.2018    source источник
comment
@ADM Я видел этот пост, когда пытался исследовать свою проблему, но на самом деле я не нашел этот ответ применимым к моему сценарию.   -  person Omar Carino    schedule 21.10.2018
comment
Значит, вы не поняли там ответов. Просмотреть все представления адаптера имеют свойство Recycling. Как только ваше представление выходит из размера AdapterView, оно перерабатывается и создается снова, когда доступно для отображения на экране во время прокрутки. Вам нужно сохранить текст во время прокрутки. На мой взгляд, я бы не стал использовать EditText внутри ListView и искал какой-то другой подход (может быть ввод с помощью диалога).   -  person ADM    schedule 21.10.2018
comment
@ADM Я ценю, что ты объяснил, что я посмотрю еще раз! Спасибо!   -  person Omar Carino    schedule 21.10.2018


Ответы (1)


Вы можете добавить прослушиватель onFocusChanged для EditText, и если фокус не с EditText, обновите БД или сохраните значение для последующего поиска.

Что-то вроде: -

    et_childname.setOnFocusChangeListener(new View.OnFocusChangeListener() {
        @Override
        public void onFocusChange(View view, boolean b) {
            if(!b) {
                Child c = (Child) view.getTag();
                c.setName(et_childname.getText().toString());
                ((MainActivity)mActivity).changedChild(c); 
            }
        }
    });
  • где et_childname — это EditText
  • б, второй параметр b имеет значение true, если EditText имеет фокус, иначе false (отсюда и проверка).
  • В этом случае тег является базовым дочерним объектом, затем он модифицируется и отправляется в действие для обработки. Этот код находится в адаптере ListView.
person MikeT    schedule 21.10.2018