Listview输入框动态加载

Listview输入框动态加载

在写一些项目中,需要提交大量的数据,有时候同级别的信息汇总时,针对不同类型所要上传的信息点不同,这时候就要进行动态加载数据信息点。以录入建筑来打个比方,不同类型的建筑可能需要录入的信息点不同,这时根据建筑的类型做动态的信息点加载。

1.输入框类型

这里输入类型大体分为三种,editText,Spinner,TextView。这里涉及到listview的分组,所以要重写public int getItemViewType方法。

public int getItemViewType(int position) {

switch (lists.get(position).getType())

{

case "input":

return 0;

case "select":

return 1;

case "date":

return 2;

default:

return 0;

}

}


2.spinner类型

这里需要把 spinner所选的值保存到form_spinner对象中去。这样做有两点原因,其一是listview滚动会进行重复加载,重新调用getview方法,将form_spinner的值赋值到spinner中去,如果不进行保存下拉框的值,下拉框将会刷新成初始值。其二,在将spinner中的值提交时,如果使用findviewbyid的方法去获取spinner的话,是无法得到屏幕外的spinner的,取值时还要从form_pinner中获取。

case 1:

final Form form_spinner = (Form) getItem(position);

if (convertView ==null) {

viewHolder_select =new ViewHolder_select();

convertView = View.inflate(context, R.layout.spinner_item,null);

viewHolder_select.textView = convertView.findViewById(R.id.spinner_name);

viewHolder_select.spinner = convertView.findViewById(R.id.form_spinner);

convertView.setTag(viewHolder_select);

}else {

viewHolder_select = (ViewHolder_select) convertView.getTag();

}

viewHolder_select.textView.setText(form_spinner.getName());

ArrayList arrayList =new ArrayList<>();

try {

if (!form_spinner.getList().equals("[]")) {

JSONArray array_item =new JSONArray(form_spinner.getList());

arrayList.clear();

arrayList.add(new Option("","请选择"));

for (int j =0; j < array_item.length(); j++) {

JSONObject object_item = array_item.getJSONObject(j);

String item = object_item.getString("DIC_VALUE");

arrayList.add(new Option(item, item));

}

}

}catch (JSONException e) {

e.printStackTrace();

}

ArrayAdapter adapter =new ArrayAdapter(context, android.R.layout.simple_spinner_item, arrayList);

adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);

viewHolder_select.spinner.setAdapter(adapter);

int select_position =0;

for (Option option : arrayList) {

if (option.getValue().equals(form_spinner.getValue())) {

select_position = adapter.getPosition(option);

}

}

viewHolder_select.textView.setText(form_spinner.getName());

//                viewHolder_select.spinner.set

                viewHolder_select.spinner.setSelection(select_position,true);

viewHolder_select.spinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {

@Override

                    public void onItemSelected(AdapterView parent, View view,int item_position,long id) {

Option option = (Option)viewHolder_select.spinner.getItemAtPosition(item_position);

String newValue = option.getValue();

form_spinner.setValue(newValue);

}

@Override

                    public void onNothingSelected(AdapterView parent) {

}

});

break;

3.Textview类型

在这里使用Textview,主要是为了填写日期时使用日期控件填写。

case 2:

final Form form_date = (Form) getItem(position);

if (convertView ==null) {

viewHolder_date =new ViewHolder_date();

convertView = View.inflate(context, R.layout.time_item,null);

viewHolder_date.textView = convertView.findViewById(R.id.date_name);

viewHolder_date.editableText = convertView.findViewById(R.id.date_value);

convertView.setTag(viewHolder_date);

}else {

viewHolder_date = (ViewHolder_date) convertView.getTag();

}

viewHolder_date.textView.setText(form_date.getName());

if (TextUtils.isEmpty(form_date.getValue())) {

viewHolder_date.editableText.setTextKeepState("");

}else {

viewHolder_date.editableText.setTextKeepState(form_date.getValue());

}

viewHolder_date.editableText.setOnClickListener(new View.OnClickListener() {

@Override

        public void onClick(View v) {

DatePickerDialog.OnDateSetListener listener =new DatePickerDialog.OnDateSetListener() {

@Override

                public void onDateSet(DatePicker arg0,int year,int month,int day) {

viewHolder_date.editableText.setText(year +"-" + (++month) +"-" + day);//将选择的日期显示到TextView中,因为之前获取month直接使用,所以不需要+1,这个地方需要显示,所以+1

                    form_date.setValue(year +"-" + (++month) +"-" + day);

}

};

DatePickerDialog dialog =new DatePickerDialog(context,0, listener,2019,1,1);//后边三个参数为显示dialog时默认的日期,月份从0开始,0-11对应1-12个月

            dialog.show();

}

});

break; 

4.EditText类型

使用listview中带有EditText控件会出现很多的问题。

其一:为了把editText改变后的值保存到form中以防止滚动时重复调用getview导致输入框文本混乱,需要使用TextWatcher来监听EditText中的文本更改。使用addTextChangedListener方法来给EditText设置文本变化监听,但在设置EditText的文本前,一定要调用removeTextChangedListener方法,保证EditText的监听列表中只有一个TextWatcher对象。如果调用此方法,会造成数据的紊乱。

其二:在滑动屏幕时,用于getview重复调用会造成数据的紊乱。

case 0:

final Form form = (Form) getItem(position);

if (convertView ==null) {

viewHolder =new ViewHolder();

convertView = View.inflate(context, R.layout.form_item,null);

viewHolder.textView = convertView.findViewById(R.id.form_name);

viewHolder.editText = convertView.findViewById(R.id.form_value);

convertView.setTag(viewHolder);

}else {

viewHolder = (ViewHolder) convertView.getTag();

}

//判断文本输入类型

                if (form.getFont().equals("number")) {

viewHolder.editText.setInputType(InputType.TYPE_CLASS_NUMBER | InputType.TYPE_NUMBER_FLAG_DECIMAL);

}

viewHolder.textView.setText(form.getName());

//步骤1:删除步骤5中添加的android.text.TextWatcher以确保android.text.TextWatcher

//不要在步骤2中触发;

//为什么?

//

//注意:当某个类型的对象附加到可编辑状态时,

//当EidtText的文本发生变化时,将调用TextWatcher的方法。

//

// EditText使用ArrayList 类型对象来存储监听器,所以我们必须这样做

//确保此列表中只有一个TextWatcher对象;

//

//避免在第2步触发TextWatcher的方法,我们会在第一时间删除它。

                if (viewHolder.editText.getTag()instanceof TextWatcher) {

viewHolder.editText.removeTextChangedListener((TextWatcher) (viewHolder.editText.getTag()));

}

//步骤2:删除android.text.TextWatcher后设置文本和焦点(步骤1);

                viewHolder.editText.setHint("请填写信息");

// set text

                if (TextUtils.isEmpty(form.getValue())) {

viewHolder.editText.setTextKeepState("");

}else {

viewHolder.editText.setTextKeepState(form.getValue());

}

//设置焦点状态

//为什么?

//

//注意:ListView有一个非常优雅的回收算法。 因此ListView中的视图不可靠。

//特别是在这种情况下,EditText是ListView的一个项目。 软件输入窗口可能会导致

// ListView relayout领先适配器的getView()多次调用。

//最重要的是,如果我们直接在EditText级别(而不是在适配器中)更改EditText的焦点状态。

//当特殊视图在其他位置重复使用时,焦点状态可能会混乱。

//

//所以使用数据源控件View的状态是处理这个问题的核心。

                if (form.isFocus()) {

if (!viewHolder.editText.isFocused()) {

viewHolder.editText.requestFocus();

}

CharSequence text = form.getValue();

// reset cursor.

                    viewHolder.editText.setSelection(TextUtils.isEmpty(text) ?0 : text.length());

}else {

if (viewHolder.editText.isFocused()) {

viewHolder.editText.clearFocus();

}

}

//步骤3:将OnTouchListener设置为EditText以更新数据源中的焦点状态指示器

//为什么?

//

//在第2步中,我们知道必须通过数据源控制视图状态。 我们使用OnTouchListener

//观察状态变化并在用户向上移动手指时更新数据源(ACTION_UP)。

//我们不想使用touch事件,只需在onTouch()方法中返回false。

                viewHolder.editText.setOnTouchListener(new View.OnTouchListener() {

@Override

                    public boolean onTouch(final View v, MotionEvent event) {

if (event.getAction() == MotionEvent.ACTION_UP) {

final boolean focus =form.isFocus();

check(position);

if (!focus && !viewHolder.editText.isFocused()) {

viewHolder.editText.requestFocus();

viewHolder.editText.onWindowFocusChanged(true);

}

}

return false;

}

});

//步骤4:将TextWatcher设置为EditText以监听EditText中的文本更改以更新数据源中的文本

                final TextWatcher textWatcher =new TextWatcher() {

@Override

                    public void beforeTextChanged(CharSequence s,int start,int count,int after) {

}

@Override

                    public void onTextChanged(CharSequence s,int start,int before,int count) {

}

@Override

                    public void afterTextChanged(Editable s) {

if (TextUtils.isEmpty(s)) {

form.setValue("");

}else {

form.setValue(String.valueOf(s));

}

}

};

viewHolder.editText.addTextChangedListener(textWatcher);

viewHolder.editText.setTag(textWatcher);

}

全部的代码在https://github.com/qwe7839056/EditTemplet

你可能感兴趣的:(Listview输入框动态加载)