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