Android通过listview实现输入框自定义提示栏(代替AutoCompleteTextView自动完成文本框)

效果图

Android通过listview实现输入框自定义提示栏(代替AutoCompleteTextView自动完成文本框)_第1张图片

背景

本人因为一些需求初次接触android,需要实现一个类似android自带的AutoCompleteTextView(自动完成文本框),但和其不同的是通过后端接口直接筛选数据(自己的分词处理规则),然后返回前端直接显示即可。

思路

 这个listview“提示框”在输入框获得焦点的情况下才显示,并且每次输入框内内容变化时,触发网络请求更新“提示框”显示的内容,在用户点击了“提示框”中的item后,将item的内容替换输入框当前内容,并关闭提示框。

实现

第一步、配置布局文件

利用 RelativeLayout布局,让edittext和listview重叠显示,然后利用

android:layout_below="@+id/et_skin"

让listview定位在edittext输入框的底部,并通过

android:visibility="gone"

默认提示框先不显示,然后在父布局中使用

android:focusable="true"
android:focusableInTouchMode="true"

让输入框不会自动触发焦点,最后只需要等到输入框触发焦点时再显示提示框即可(这里会存在一个弊端,他这个listview布局与其他组件布局不会重叠显示,是直接顺序显示,如果有大佬知道如何优化,可以留下你的方法)。效果图及代码如下:

Android通过listview实现输入框自定义提示栏(代替AutoCompleteTextView自动完成文本框)_第2张图片

            

                

                

            

 第二步、绑定布局、适配器-实现功能

这里通过分别监听:输入框焦点获取和失去、输入框内容变化、提示框item点击事件来实现我们的需求,具体代码及备注如下:

    private List auto_list_skin = new ArrayList<>(); // 输入框自动提示内容集合
    private ArrayAdapter adapter_skin; // 适配器
    private EditText et_skin; // 输入框
    private ListView lv_order_skin; // listview提示框

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

        // 绑定组件
        et_skin = view_default.findViewById(R.id.et_skin);
        lv_order_skin = view_default.findViewById(R.id.lv_order_skin);

        // listview提示框-皮肤名-绑定数据集合
        adapter_skin = new ArrayAdapter<>(OrderActivity.this,android.R.layout.simple_list_item_1, auto_list_skin);
        // 组件绑定适配器
        lv_order_skin.setAdapter(adapter_skin);

        // 监听iten点击事件
        lv_order_skin.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView adapterView, View view, int i, long l) {
                et_skin.setText(auto_list_skin.get(i)); // 选中item-更新数据到输入框中
                et_skin.setSelection(auto_list_skin.get(i).length()); // 让输入框内的光标定位到最后一位
                lv_order_skin.setVisibility(View.GONE); // 隐藏提示栏
            }
        });

        // 输入框-皮肤-焦点监听事件
        et_skin.setOnFocusChangeListener(new View.OnFocusChangeListener() {
            @Override
            public void onFocusChange(View view, boolean hasFocus) {
                if (hasFocus) {

                } else {
                    lv_order_skin.setVisibility(View.GONE); // 隐藏提示栏
                }
            }
        });

        // 输入框-皮肤-输入变化事件
        et_skin.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) {
                searchInfo(et_skin.getText().toString(), adapter_skin); // 执行网络请求-更新提示框内容
                lv_order_skin.setVisibility(View.VISIBLE); // 显示提示栏
            }

            @Override
            public void afterTextChanged(Editable editable) {

            }
        });
    }

最后在网路请求结果中,通过如下代码更新数据到提示框中,这样完整的功能就实现了!

auto_list_skin.clear(); // 清空数据数组
auto_list_skin.add("新数据item"); // 添加数据到数组中
adapter_skin.notifyDataSetChanged(); // 更新数组到listtview提示框中
                        

你可能感兴趣的:(android,android)