zoukankan      html  css  js  c++  java
  • Android进阶之绘制-自定义View完全掌握(三)

    自定义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

  • 相关阅读:
    2019西湖论剑网络安全技能大赛(大学生组)部分WriteUp
    C++学习(四十)(C语言部分)之 学生管理系统设计
    C++学习(三十九)(C语言部分)之 游戏项目(2048游戏)
    C++学习(三十八)(C语言部分)之 排序(冒泡 选择 插入 快排)
    C++学习(三十七)(C语言部分)之 链式栈(推箱子实现)
    C++学习(三十六)(C语言部分)之 链表2
    C++学习(三十五)(C语言部分)之 单链表
    C++学习(三十四)(C语言部分)之 链表
    C++学习(三十三)(C语言部分)之 队列
    C++学习(三十二)(C语言部分)之 栈
  • 原文地址:https://www.cnblogs.com/blizzawang/p/11411762.html
Copyright © 2011-2022 走看看