Android進階之繪製-自定義View完全掌握(三)

  • 2020 年 1 月 20 日
  • 筆記

自定義View系列的第三篇部落格,我們來學習如何實現自定義下拉框。 今天的程式,我們來實現這樣的一個效果。

布局非常簡單,我們直接開始編碼。 修改activity_main.xml文件的程式碼。

<?xml version="1.0" encoding="utf-8"?>  <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"      xmlns:app="http://schemas.android.com/apk/res-auto"      xmlns:tools="http://schemas.android.com/tools"      android:layout_width="match_parent"      android:layout_height="match_parent"      tools:context="com.itcast.test0430.MainActivity">        <EditText          android:id="@+id/et_input"          android:layout_width="match_parent"          android:layout_height="wrap_content"          android:ellipsize="middle"          android:hint="請輸入內容..."          android:paddingRight="40dp"          android:singleLine="true" />        <ImageView          android:id="@+id/iv_down_arrow"          android:layout_width="30dp"          android:layout_height="30dp"          android:layout_alignRight="@id/et_input"          android:layout_alignTop="@id/et_input"          android:padding="5dp"          android:src="@drawable/down_arrow" />  </RelativeLayout>

布局程式碼非常簡單,就是兩個控制項。 接下來修改MainActivity的程式碼。

package com.itcast.test0430;    import android.graphics.Color;  import android.os.Bundle;  import android.support.v7.app.AppCompatActivity;  import android.view.View;  import android.view.ViewGroup;  import android.widget.AdapterView;  import android.widget.BaseAdapter;  import android.widget.EditText;  import android.widget.ImageView;  import android.widget.ListView;  import android.widget.PopupWindow;  import android.widget.TextView;    import java.util.ArrayList;    import butterknife.BindView;  import butterknife.ButterKnife;  import butterknife.OnClick;    public class MainActivity extends AppCompatActivity {        @BindView(R.id.et_input)      EditText etInput;      @BindView(R.id.iv_down_arrow)      ImageView ivDownArrow;        /**       *       */      private PopupWindow popupWindow;      private ListView listView;        private ArrayList<String> msgs;      private MyAdapter adapter;        @Override      protected void onCreate(Bundle savedInstanceState) {          super.onCreate(savedInstanceState);          setContentView(R.layout.activity_main);          ButterKnife.bind(this);            listView = new ListView(this);          listView.setBackgroundColor(Color.WHITE);          //準備數據          msgs = new ArrayList<>();          for(int i = 0;i < 500;i++) {              msgs.add(i + "--aaaaaa---" + i);          }          adapter = new MyAdapter();          listView.setAdapter(adapter);          listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {              @Override              public void onItemClick(AdapterView<?> parent, View view, int position, long id) {                  //1、得到數據                  String msg = msgs.get(position);                  //2、設置到輸入框                  etInput.setText(msg);                    if(popupWindow != null && popupWindow.isShowing()){                      popupWindow.dismiss();                      popupWindow = null;                  }              }          });      }        @OnClick(R.id.et_input)      public void onViewClick(View view){          if(popupWindow == null){              popupWindow = new PopupWindow(this);              popupWindow.setWidth(etInput.getWidth());              popupWindow.setHeight(400);                popupWindow.setContentView(listView);              popupWindow.setFocusable(true);//設置焦點          }            popupWindow.showAsDropDown(etInput,0,0);      }        class MyAdapter extends BaseAdapter{            @Override          public int getCount() {              return msgs.size();          }            @Override          public Object getItem(int position) {              return null;          }            @Override          public long getItemId(int position) {              return 0;          }            @Override          public View getView(int position, View convertView, ViewGroup parent) {              ViewHolder viewHolder;              if(convertView == null){                  convertView = View.inflate(MainActivity.this,R.layout.item_main,null);                  viewHolder = new ViewHolder();                  viewHolder.tv_msg = convertView.findViewById(R.id.tv_msg);                  viewHolder.iv_delete = convertView.findViewById(R.id.iv_delete);                  convertView.setTag(viewHolder);              }else{                  viewHolder = (ViewHolder) convertView.getTag();              }              //根據位置得到數據              final String msg = msgs.get(position);              viewHolder.tv_msg.setText(msg);                //設置刪除              viewHolder.iv_delete.setOnClickListener(new View.OnClickListener() {                  @Override                  public void onClick(View v) {                      //1、從集合刪除                      msgs.remove(msg);                      //2、刷新UI---也就是 刷新適配器                      adapter.notifyDataSetChanged();                  }              });              return convertView;          }      }      static class ViewHolder{          TextView tv_msg;          ImageView iv_delete;      }  }

item_main.xml文件的程式碼如下。

<?xml version="1.0" encoding="utf-8"?>  <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"      android:layout_width="match_parent"      android:layout_height="50dp"      android:gravity="center_vertical"      android:orientation="horizontal"      android:padding="5dp">        <ImageView          android:layout_width="50dp"          android:layout_height="50dp"          android:layout_gravity="center_vertical"          android:layout_margin="5dp"          android:padding="3dp"          android:src="@drawable/user" />        <TextView          android:id="@+id/tv_msg"          android:layout_width="wrap_content"          android:layout_height="wrap_content"          android:layout_gravity="center_vertical"          android:layout_margin="5dp"          android:layout_weight="1"          android:gravity="center"          android:padding="3dp"          android:text="三個火槍手"          android:textColor="#000" />        <ImageView          android:id="@+id/iv_delete"          android:layout_width="30dp"          android:layout_height="30dp"          android:layout_gravity="center_vertical"          android:layout_margin="5dp"          android:padding="3dp"          android:src="@drawable/delete" />  </LinearLayout>

這裡的程式碼也很簡單,所以不作過多解釋,程式碼的每個地方都有注釋。唯一需要注意的地方就是,因為我們的PopupWindow類是設置了寬為200,而只要是在程式碼中設置的控制項屬性,它的單位均為px(像素),而像素是沒有適配功能的,所以為了使我們的程式能夠在任意解析度的手機上正確運行,我們應該把像素轉換為dp。 提供給大家一個工具類,用於dp與px之間的轉換。

package com.itcast.test0430;    import android.content.Context;    public class DensityUtil {      /**       * 根據手機的解析度從dip的單位轉換為px(像素)       */      public static int dipToPx(Context context,float dpValue){          final float scale = context.getResources().getDisplayMetrics().density;          return (int) (dpValue * scale + 0.5f);      }        /**       * 根據手機的解析度從px(像素)的單位轉換為dip       */      public static int pxToDip(Context context,float pxValue){          final float scale = context.getResources().getDisplayMetrics().density;          return (int) (pxValue * scale + 0.5f);      }  }

所以,我們把popupWindow.setHeight(400);改為

int height = DensityUtil.dipToPx(this,400);  popupWindow.setHeight(height);

即可。 現在運行項目,預覽一下效果。

這樣,我們的下拉框也就實現了。現在有了dp和px之間轉換的工具類,我們就可以在需要螢幕適配的地方使用它了,包括我們之前練習的一些項目。

源碼已上傳至GitHub