Dialog自定义详解(一)

引言

最近公司做了一个电商的项目,运用到了非常多的Dialog自定义 ,在这里进行一下总结,以便于后面查阅资料。

如何使用原生的Dialog

下面就是展示原生的android代码:

private void showNormalDialog() {
        /* @setIcon 设置对话框图标
         * @setTitle 设置对话框标题
         * @setMessage 设置对话框消息提示
         * setXXX方法返回Dialog对象,因此可以链式设置属性
         */
        final AlertDialog.Builder normalDialog =
                new AlertDialog.Builder(DialogActivity.this);
        normalDialog.setIcon(R.mipmap.ic_launcher)
                .setTitle("我是一个普通Dialog")
                .setMessage("你要点击哪一个按钮呢?")
                .setPositiveButton("确定",
                        new DialogInterface.OnClickListener() {
                            @Override
                            public void onClick(DialogInterface dialog, int which) {
                                //...To-do
                            }
                        })
                .setNegativeButton("关闭",
                        new DialogInterface.OnClickListener() {
                            @Override
                            public void onClick(DialogInterface dialog, int which) {
                                //...To-do
                            }
                        });
        // 显示
        normalDialog.show();
    }

但是大家有木有感觉原生的样式丑爆了,所以自定义弹框是每个android程序员的必修课。
自定义弹框是非常简单的,我们该怎么写出通用,并且好用的弹框,才是我们应该学习的。

模拟原生的Dialog写自己喜欢的Dialog

下面我们来进入今天的实用性场景,一般开发中我们经常会出现,需要展示一些描述性的东西,提示完后你需要点选否或者是,我们来写一个关于这种简单类型的自定义Dialog:

写DescribeDialog的布局_dialog_describe
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:id="@+id/root"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:background="@android:color/white"
              android:orientation="vertical">
    <!-- 描述内容 -->
    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <TextView
            android:id="@+id/tv_describe"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:gravity="center"
            android:maxEms="13"
            android:padding="10dp"
            android:text="这是一个描述型自定义Dialog,起到了提示的作用,
            它一行最大的字数显示在13位,maxEms起到的作用。"
            android:textSize="17sp"/>
    </RelativeLayout>

    <View
        android:layout_width="match_parent"
        android:layout_height="1dp"
        android:background="#e6e6e6"/>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="45dp"
        android:orientation="horizontal">
        <!-- 取消按钮 -->
        <TextView
            android:id="@+id/tv_cancel"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            android:gravity="center"
            android:text="取消"
            android:textColor="@android:color/holo_red_dark"
            android:textSize="17sp"/>

        <View
            android:layout_width="1dp"
            android:layout_height="match_parent"
            android:background="#e6e6e6"/>
        <!-- 确定按钮 -->
        <TextView
            android:id="@+id/tv_sumit"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            android:gravity="center"
            android:text="确定"
            android:textSize="17sp"/>
    </LinearLayout>
</LinearLayout>
写DescribeDialog的类

因为属性扩展性的自定义我们的DescribeDialog直接继承Dialog
为了跟原生的使用方法一样,我在这里定义了Builder运用建造者模式来注入参数,我们先来看下DescribeDialogParams,也就是参数类:

  **
     * 参数类
     */
    private static class DescribeDialogParams{
        //上下文对象
        public Context mContext;
        //布局扩展器
        public LayoutInflater mInflater;


        //描述的内容
        public CharSequence describe;
        //提交按钮展示的内容
        public CharSequence sumit;
        //取消按钮展示内容
        public CharSequence cancel;

        //描述的字体颜色
        public int describeTextColor=-1;
        //提交按钮字体颜色
        public int sumitTextColor=-1;
        //取消按钮字体颜色
        public int cancelTextColor=-1;

        //是否能取消 默认是 能
        public boolean mCancelable;

        //确定按钮监听器
        public OnClickListener mPositiveButtonListener;
        //取消按钮监听器
        public OnClickListener mNegativeButtonListener;

        public DescribeDialogParams(Context context){
            mContext=context;
            mCancelable=true;
            mInflater= (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        }

    }

注释写的很清楚了,这个类就是用来保持参数的引用,这在很多参数的时候是非常有必要的,这种方式应用于各种开源项目中,下面来看下建造者:

 /**
     * 建造者
     */
    public static class Builder {
        private  DescribeDialogParams P;
        private int mTheme;


        public Builder(@NonNull Context context) {
            this(context,R.style.NoTitleDialog);
        }
        public Builder(@NonNull Context context,int themeResId) {
            P=new DescribeDialogParams(context);
            this.mTheme=themeResId;
        }

        public Context getContext() {
            return P.mContext;
        }

        /**
         *
         * @param describe 设置描述的内容
         * @return
         */
        public Builder setDescribe(CharSequence describe){
            P.describe=describe;
            return this;
        }

        /**
         *
         * @param sumit 设置提交按钮的展示内容
         * @param onClickListener 提交按钮的监听器
         * @return
         */
        public Builder setPositiveButton(CharSequence sumit,OnClickListener onClickListener){
            P.sumit=sumit;
            P.mPositiveButtonListener=onClickListener;
            return this;
        }

        /**
         *
         * @param cancel 设置取消按钮的展示内容
         * @param onClickListener 取消按钮的监听器
         * @return
         */
        public Builder setNegativeButton(CharSequence cancel,OnClickListener onClickListener){
            P.cancel=cancel;
            P.mNegativeButtonListener=onClickListener;
            return this;
        }
        /**
         *
         * @param describeTextColor 设置描述内容的字体颜色
         * @return
         */
        public Builder setDescribeTextColor(int describeTextColor){
            P.describeTextColor=describeTextColor;
            return this;
        }

        /**
         *
         * @param sumitTextColor 设置提交按钮的字体颜色
         * @return
         */
        public Builder setSumitTextColor(int sumitTextColor){
            P.sumitTextColor=sumitTextColor;
            return this;
        }

        /**
         *
         * @param cancelTextColor 设置取消按钮的字体颜色
         * @return
         */
        public Builder setCancelTextColor(int cancelTextColor){
            P.cancelTextColor=cancelTextColor;
            return this;
        }

        /**
         * 创建DescribeDialog
         * @return DescribeDialog
         */
        public DescribeDialog create() {
            final DescribeDialog dialog = new DescribeDialog(P.mContext,mTheme);
            dialog.setP(P);
            return dialog;
        }

        /**
         *展示DescribeDialog
         * @return DescribeDialog
         */
        public DescribeDialog show() {
            final DescribeDialog dialog = create();
            dialog.show();
            return dialog;
        }
    }

建造者也比较简单,就是参数的注入,然后返回当前建造的对象,这种对于参数多的情况下链式结构注入参数,还有就是有必要提醒的是,show()方法必须在create()之后执行,之前注入的参数才有效。
我们再来看看DescribeDialog类:

   private Context mContext;
    private TextView tvDescribe;
    private TextView tvCancel;
    private TextView tvSumit;
    private DescribeDialog(Context context, int themeResId) {
        super(context, themeResId);
        this.mContext=context;
    }

    private void setP(DescribeDialogParams p) {
        P = p;
    }

    private  DescribeDialogParams P;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        init();
    }

    private void init() {
        View view = P.mInflater.inflate(R.layout.dialog_describe, null);
        setContentView(view);
        Window dialogWindow = getWindow();
        WindowManager.LayoutParams lp = dialogWindow.getAttributes();
        DisplayMetrics d = mContext.getResources().getDisplayMetrics(); // 获取屏幕宽、高用
        lp.width = (int) (d.widthPixels * 0.8); // 高度设置为屏幕的0.6
        dialogWindow.setAttributes(lp);
        //点击外部不关闭
        setCancelable(P.mCancelable);

        //初始化View
        tvDescribe= (TextView) view.findViewById(R.id.tv_describe);
        tvCancel= (TextView) view.findViewById(R.id.tv_cancel);
        tvSumit= (TextView) view.findViewById(R.id.tv_sumit);
        //设置颜色
        if (P.describeTextColor!=-1){
            tvDescribe.setTextColor(P.describeTextColor);
        }
        if (P.cancelTextColor!=-1){
            tvCancel.setTextColor(P.cancelTextColor);
        }
        if (P.sumitTextColor!=-1){
            tvSumit.setTextColor(P.sumitTextColor);
        }
        //设置内容显示
        if (!TextUtils.isEmpty(P.describe)){
            tvDescribe.setText(P.describe);
        }
        if (!TextUtils.isEmpty(P.cancel)){
            tvCancel.setText(P.cancel);
        }
        if (!TextUtils.isEmpty(P.sumit)){
            tvSumit.setText(P.sumit);
        }
        //监听器
        tvSumit.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                dismiss();
                if (P.mPositiveButtonListener!=null){
                    P.mPositiveButtonListener.onClick(tvSumit.getId());
                }
            }
        });
        tvCancel.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                dismiss();
                if (P.mNegativeButtonListener!=null){
                    P.mNegativeButtonListener.onClick(tvSumit.getId());
                }
            }
        });
    }

大家看代码,可以看到,DescribeDialog类有DescribeDialogParams的依赖,当我们需要参数的值的时候直接从DescribeDialogParams中取,这种建造者模式就是我们自定义Dialog的精髓所在,然后最重要的方法就在init()方法中,对View的扩展,使用布局扩展器扩展出View后通过setContentView(view)来设置到Dialog中,有木有感觉Dialog就是一个Activity,之后就是一些对Dialog宽高的设置。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,440评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,814评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,427评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,710评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,625评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,014评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,511评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,162评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,311评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,262评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,278评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,989评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,583评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,664评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,904评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,274评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,856评论 2 339

推荐阅读更多精彩内容