Android中控制元件AutoCompleteTextView的使用方法和一些屬性
AutoCompleteTextView一些屬性
<!--completionThreshold:它的值決定了你在AutoCompleteTextView至少輸入幾個字元,它才會具有自動提示的功能--> <!--dropDownWidth設定提示框的寬度--> <!--completionHintView這是框框底部顯示的文字,也可以用completionHint--> <!--dropDownSelector 提示框中點選時的背景顏色--> <!--dropDownVerticalOffset 距搜尋框的垂直距離--> <!--popupBackground 設定下方提示框背景 @null表示透明--><!--android:background="@null" 用去去掉AutoCompleteTextView本身自帶的下劃線-->
這是我總結的AutoCompleteTextView特有的屬性,因為AutoCompleteTextView的父類是EditText,而EditText的父類是TextView,所以AutoCompleteTextView除了自己特有的屬性之外,EditText和TextView的所有屬性都已引用。
AutoCompleteTextView使用方法
關於 AutoCompleteTextView 的使用分為三步:
一、首先要在佈局中引用該控制元件,
二 、java檔案中例項化物件,
三、設定介面卡,可以自定義介面卡,也可以直接使用預設的系統提供的介面卡,自定義介面卡的話可能比較複雜一些,但是不是很難,看過一遍程式碼後,就能馬上搞懂。
下面我將貼出我的程式碼
這是我在佈局中寫的程式碼
<AutoCompleteTextView android:id="@+id/auto_seekTips" android:layout_width="match_parent" android:layout_height="35dp" android:layout_marginLeft="20dp" android:layout_marginRight="20dp" android:layout_marginTop="5dp" android:background="@drawable/seekback" android:completionThreshold="1" android:drawableLeft="@drawable/seek16" android:dropDownWidth="match_parent" android:dropDownVerticalOffset="10dp" android:hint="請輸入城市名稱" android:inputType="text" android:lines="1" android:maxLength="20" android:paddingLeft="15dp" android:textSize="15sp" />
這是我在 MainActivity寫的程式碼
//搜尋提示控制元件
private AutoCompleteTextView auto_seektips;
//將控制元件例項化
auto_seektips = findViewById(R.id.auto_seekTips);
自定義介面卡的設定
AutoCompleteTextAdapter adapter=new AutoCompleteTextAdapter(resultsList, this);
auto_seektips.setAdapter(adapter);
AutoCompleteAdapter文字內容變化前 ,變化時,變化後的一個實時監聽
//AutoCompleteTextView控制元件的點選事件
auto_seektips.addTextChangedListener(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) {
//判斷搜尋框中是否有內容,然後再決定其下方的熱門城市的是否隱藏
//android提供了一個專門判斷空字串的方法:TextUtils.isEmpty
if (TextUtils.isEmpty(auto_seektips.getText())||auto_seektips.getText()!=null) {
tv_hotCityText.setVisibility(View.GONE);
gridViewhotCity.setVisibility(View.GONE);
} else {
tv_hotCityText.setVisibility(View.VISIBLE);
gridViewhotCity.setVisibility(View.VISIBLE);
}
}
@Override//改變後
public void afterTextChanged(Editable s) {
if (!TextUtils.isEmpty(auto_seektips.getText())) {
tv_hotCityText.setVisibility(View.GONE);
gridViewhotCity.setVisibility(View.GONE);
} else {
tv_hotCityText.setVisibility(View.VISIBLE);
gridViewhotCity.setVisibility(View.VISIBLE);
}
}
});
AutoCompleteAdapter點選事件的設定,
auto_seektips.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
auto_seektips.setText("");
//獲取到過濾後的結果的集合
ArrayList<Result> results=AutoCompleteTextAdapter.newvalues;
Result result=results.get(position);
String s=result.getDistrict();
auto_seektips.setText(s);
//將游標放到文字最後
auto_seektips.setSelection(auto_seektips.getText().length());
}
});
設定介面卡
這是我自定義的介面卡
public class AutoCompleteTextAdapter extends BaseAdapter implements Filterable {
private ArrayFilter mFilter;//資料過濾器
private List<Result> mList;//傳進來的資料
private ArrayList<Result> mFilteredData;//
private Context context;//上下文
public static ArrayList<Result> newvalues;
public AutoCompleteTextAdapter(List<Result> mList, Context context) {
this.mList = mList;
this.context = context;
}
@Override
public int getCount() {
return mList == null ? 0 : mList.size();
}
@Override
public Object getItem(int position) {
return mList.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View view;
ViewHolder viewHolder;
if (convertView == null) {
view = View.inflate(context, R.layout.seektipitem2, null);
viewHolder = new ViewHolder();
viewHolder.tv_province = view.findViewById(R.id.tv_Privance);
viewHolder.tv_city = view.findViewById(R.id.tv_City);
viewHolder.tv_district = view.findViewById(R.id.tv_auto_seektips);
view.setTag(viewHolder);
} else {
view = convertView;
viewHolder = (ViewHolder) view.getTag();
}
Result resultlist = mList.get(position);
viewHolder.tv_province.setText(resultlist.getPrivance());
viewHolder.tv_city.setText(resultlist.getCity());
viewHolder.tv_district.setText(resultlist.getDistrict());
//判斷:如果市和縣的內容一樣,則讓市的控制元件消失
if (resultlist.getDistrict() == resultlist.getCity()) {
viewHolder.tv_city.setVisibility(View.GONE);
}
return view;
}
class ViewHolder {
public TextView tv_province;
public TextView tv_city;
public TextView tv_district;
}
/**
* 在後臺執行緒執行,定義過濾演算法
* getFilter()方法會返回一個Filter物件,Filter是一個數據過濾器,其過濾操作是通過performFiltering()方法和publishResult()方法完成的。
* performFiltering()方法進行過濾操作,publishResult()方法用於發表過濾操作結果。
*
* @return
*/
@Override
public Filter getFilter() {
//建立過濾器的物件
if (mFilter == null) {
mFilter = new ArrayFilter();
}
return mFilter;
}
//過濾器類
class ArrayFilter extends Filter {
/**
* @param constraint:使用者在輸入框中所輸入的內容
* @return
*/
@Override//進行過濾操作
protected FilterResults performFiltering(CharSequence constraint) {
//使用過濾操作的結果
FilterResults results = new FilterResults();
if (mFilteredData == null) {
//以一個集合初始化mFilteredData=new ArrayList<String>(mList);//mList(陣列容量)為集合
mFilteredData = new ArrayList<>(mList);
}
//如果沒有輸入內容,則不過濾
if (constraint == null || constraint.length() == 0) {
ArrayList<Result> list = mFilteredData;
results.values = list;
results.count = list.size();
} else {
//過濾的條件
String constraintString = constraint.toString().toLowerCase();
//將傳進來的全部資料賦值給filteredValues
ArrayList<Result> filteredValues = mFilteredData;
int count = filteredValues.size();
newvalues = new ArrayList<>(count);
//迴圈變數資料來源,如果有屬性滿足過濾條件,則新增到result中
for (int i = 0; i < count; i++) {
Result resultData = filteredValues.get(i);
if (resultData != null) {
//過濾條件
if (resultData.getDistrict() != null && resultData.getDistrict().startsWith(constraintString)) {
newvalues.add(resultData);
}
}
results.values = newvalues;
results.count = newvalues.size();
}
}
return results;
}
@Override//發表過濾操作結果
protected void publishResults(CharSequence constraint, FilterResults results) {
mList = (List<Result>) results.values;
if (results.count > 0) {
//重繪當前可見區域
notifyDataSetChanged();
} else {
//重繪控制元件,還原到初始狀態
notifyDataSetInvalidated();
}
}
}
}
上面是我自定義的適配,我大致總結一下,首先要繼承BaseAdapter,然後還要實現Filterable介面,重寫getFilter()
,這個方法會返回一個Filter物件,這是與ListView,RelcyclerView等的列表的介面卡不太相同的地方。也要定義一個類,來繼承Filter過濾類,並且要實現performFiltering()和publishResult(),前者是進行過濾操作的,後者是用於對過濾後的結果的發表。在performFiltering()方法中,你可以根據自己的需求對過濾條件進行設定。對與BaseAdapter我就不詳細的說了,相信大家現在已經用的很熟練了。在程式碼中我也做了詳細的註釋,我也就不對每一部分進行詳細的講解了,有什麼不懂的可以給我留言。
這是系統預設的介面卡
//陣列
private String[] HotCityName = {"北京", "上海","南京","廣州", "深圳", "西安", "瀋陽", "成都", "佛山","重慶","武漢","鄭州","太原","石家莊","長春","青島"
,"天津","昆明"};
//從strings.xml檔案中獲取城市陣列
seekTipContents = getResources().getStringArray(R.array.list);
// 為AutoCompleteTextView設定介面卡
ArrayAdapter<String> arrayAdapter = new ArrayAdapter<String>(MainActivity.this,R.layout.seektipitem2, R.id.tv_auto_seektips, seekTipContents);
auto_seektips.setAdapter(arrayAdapter);
如果使用系統自帶的介面卡的話,雖然簡單易用,但是搜尋提示顯示出來的介面是比較難看,以後在公司做專案的時候,一般是不會用,因為介面難看的話,會給使用者帶來極差的體驗感
由於我做的是一個新增城市的功能,我全國所有的城市放在了本地的XML檔案中,所以下面我將貼出關於XML解析的詳細程式碼。為了與上方我寫的程式碼更加的匹配。
//省市縣的XML資料解析
public ArrayList<Result> Parserxml(XmlPullParser parser) {
ArrayList<Result> cityArray = new ArrayList<>();
Result resultTemp = null;
String province = null, city = null, district = null;
try {
int eventType = parser.getEventType();
while (eventType != XmlPullParser.END_DOCUMENT) {
switch (eventType) {
case XmlPullParser.START_DOCUMENT:
break;
case XmlPullParser.START_TAG:
//給當前的標籤起一個名字
String tagName = parser.getName();
if (tagName.equals("resources")){
}else if (tagName.equals("item")) {
resultTemp = new Result();
}else if(tagName.equals("id")){
}else if (tagName.equals("province")) {
province = parser.nextText();
} else if (tagName.equals("city")) {
city = parser.nextText();
} else if (tagName.equals("district")) {
district = parser.nextText();
resultTemp.setPrivance(province);
resultTemp.setCity(city);
resultTemp.setDistrict(district);
cityArray.add(resultTemp);
}
break;
case XmlPullParser.END_TAG:
break;
case XmlPullParser.END_DOCUMENT:
break;
}
eventType=parser.next();
}
} catch (XmlPullParserException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return cityArray;
}
我現在也是一個android開發的新手,如果哪寫的不好,希望大家可以給我指出,有什麼不懂的可以給我留言,若有什麼新的見解的,也可以給我留言,我們互相交流,共同進步。