建造者模式衍生的全局Dialog

这篇小文将讲述我是如何根据建造者设计模式来实现一个全局Dialog。
如果各位看官还不太了解建造者设计模式,建议可以看一下我的上篇文章。

背景

在每个项目当中,都会封装一些全局的样式,比如全局Loading、全局Dialog等。
封装这些功能是因为这些控件的使用频率极高。
在我刚接手项目A的时候,项目A也不例外,拥有着全局的Dialog。
刚开始时我尽量写一些侵入性低、仿照率高的代码,避免影响到之前的逻辑。并且学习着如何使用项目A的框架,这个过程我相信大家都是一样的。
当我用到Dialog的时候,我看到了项目A的全局Dialog,详细代码就不说了,给大家看一下项目A全局Dialog中的所有方法:


不知各位看官会不会被这么多的构造方法给吓到。
这些构造方法都是随着项目需求的增加而增加的。
大家都知道,一个完整Dialog包括的元素至少应该有:提示图片、标题、描述文字、按钮等。
但是这些元素都不是必须的:
在C页面我弹出的Dialog可能只要做一个温馨提示:一行描述文字外加一个确认按钮。
在D页面我可能弹出的是用户退出登录的二次确认窗口:标题+文字+两个按钮。
那么到这里,各位看官就能明白了,为什么会有那么多的构造方法。
没错,我们在new KLCustimDialog()的同时,需要把所有的参数都传入进去。
这种写法的问题以及维护成本之高,我就不做过多描述了,我就简单给大家加个需求:
要求点击Dialog外部不能取消Dialog。
想想这个需求下的构造方法,要添加几个?

基本

当我刚看到建造者模式的时候,我真是又惊又喜,热血沸腾!
我第一时间想到的就是重构项目A的全局Dialog!
大家都知道,Android中的AlertDialog就是使用建造者模式来实现的。
在我模仿构思了一波之后,创建Dialog的代码是这样的:

Dialog.Builder builder = new Dialog.Builder(context);
builder.setTitle("提示");
builder.setMessage("确认退出登录吗?");
builder.setLeftText("取消");
builder.setRightText("确认");
builder.setOnclickListener(listener);
Dialog dialog = builder.creater();
dialog.show();

这样写似乎没有什么问题了。
我们把Dialog的所有元素都默认隐藏,在调用某个元素的填充方法后,我们就将其显示出来。
这样我们就摆脱了无限多的构造方法,完美!
文章到这里就应该结束了?
怎么可能!我都没变形呢!

我见别人的Builder模式都是这样写的:

Picasso.with(context)
    .load(url)
    .fit()
    .config(XXX)
    .placeholder(xxx)
    .error(xxx)
    .into(imageView);

上述代码是使用Picasso来加载url图片。
一行代码完成。
帅不帅,想不想学?
所以我就想能不能用上面作为模板,来实现我们的全局Dialog。
话不多说,说干就干。

变形

完整代码我已经上传GitHub,看代码我还是建议各位看官去我的GitHub上看,比较整洁。
最终实现的效果如下:


首先,让我们来回想一下建造者模式的组成:

  • Product:产品角色
  • Builder:抽象的建造者
  • ConcreteBuilder:具体的建造者
  • Director:指挥者

接下来,我们再把这些成员转化为我们的全局Dialog的成员:

  • Product:Dialog就是我们制作出来的产品
  • Builder:Dialog参数拼接抽象
  • ConcreteBuilder:Dialog参数拼接细节
  • Director:所有用到该Dialog的地方都是指挥者,它们决定着Dialog具体样式。

思路有了,下面就开始动手吧,首先我们来创建Dialog的参数封装,里面应该有Dialog所有组成元素:

private static class DialogParams {
    private Context context;
    //标题
    private String title;
    //标题字体大小
    private int titleSizeSp;
    //图标资源
    private int imageResource;
    //图标宽
    private int imageWidth;
    //图标高
    private int imageHeight;
    //消息内容
    private String message1;
    //消息内容文字位置
    private int message1Gravity = Gravity.CENTER;
    //点击外部是否可以取消
    private boolean isCanCancel = true;
    //左边按钮内容
    private String leftButtonText;
    //左边按钮颜色
    private int leftBtColor;
    //左边点击事件
    private ConcreteBuilder.ButtonClickLister leftListener;
    //右边按钮内容
    private String rightButtontText;
    //右边边按钮颜色
    private int rightBtColor;
    //右边按钮点击事件
    private ConcreteBuilder.ButtonClickLister rightListener;
}

我使用了内部类去实现了整个Dialog,整个Dialog只有一个类,所以所有参数都是private并且没有提供set、get。
并且我为了方便,省略了Builder抽象类,直接构造了Builder抽象类的实现ConcreteBuilder

 public static class ConcreteBuilder {
     //持有Product对象
     private DialogParams p;
     
     ConcreteBuilder(Context context) {
         p = new DialogParams();
         p.context = context;
     }
     public ConcreteBuilder title(String text) {
         p.title = text;
         return builder;
     }
     public ConcreteBuilder titleSize(int spSize) {
         p.titleSizeSp = spSize;
         return builder;
     }
     public ConcreteBuilder imageResource(int imageResource) {
         p.imageResource = imageResource;
         return builder;
     }
     public ConcreteBuilder imageWidth(int imageWidth) {
         p.imageWidth = imageWidth;
         return builder;
     }
     public ConcreteBuilder imageHeight(int imageHeight) {
         p.imageHeight = imageHeight;
         return builder;
     }
     public ConcreteBuilder message(String text) {
         p.message1 = text;
         return builder;
     }
     public ConcreteBuilder messageGravity(int gravity) {
         p.message1Gravity = gravity;
         return builder;
     }
     public ConcreteBuilder canCancel(boolean isCanCancel) {
         p.isCanCancel = isCanCancel;
         return builder;
     }
     public ConcreteBuilder leftBt(String text, ButtonClickLister lister) {
         p.leftButtonText = text;
         p.leftListener = lister;
         return builder;
     }
     public ConcreteBuilder leftBtColor(int color) {
         p.leftBtColor = color;
         return builder;
     }
     public ConcreteBuilder rightBtColor(int color) {
         p.rightBtColor = color;
         return builder;
     }
     public ConcreteBuilder rightBt(String text, ButtonClickLister lister) {
         p.rightButtontText = text;
         p.rightListener = lister;
         return builder;
     }
     void clear() {
         p = null;
     }
     public DialogProduct create() {
         return new DialogProduct(p);
     }
     //按钮点击回调
     public interface ButtonClickLister {
         void onClick(DialogProduct dialog);
     }
 }

我们会发现每个参数拼接方法都会返回ConcreteBuilder,这里是实现一行代码构建Dialog的关键。
参考Picasso的书写方式,明显可以看出它没有进行new的行为,说明with()一定是静态的,随之with()返回的对象也必为静态。
为了实现Picasso的书写方式,我们这里也将ConcreteBuilder静态,方便实现一句话创建Dialog。
接下来就是Dialog的代码:

public class DialogProduct extends Dialog {

    private TextView tvTitle;
    private ImageView ivIcon;
    private TextView tvMessage;
    private TextView tvButtonLeft;
    private TextView tvButtonRight;
    private ImageView viewLine;

    //持有Builder
    private static ConcreteBuilder builder;

    //模仿Picasso的书写方式
    public static ConcreteBuilder with(Context context) {
        if (builder == null) {
            builder = new ConcreteBuilder(context);
        }
        return builder;
    }


    private DialogProduct(DialogParams p) {
        //设置没有标题的Dialog风格
        super(p.context, R.style.NoTitleDialog);

        View contentView = LayoutInflater.from(p.context).inflate(R.layout.dialog_build, null);
        setContentView(contentView);

        tvTitle = contentView.findViewById(R.id.tv_title);
        ivIcon = contentView.findViewById(R.id.iv_icon);
        tvMessage = contentView.findViewById(R.id.tv_message);
        tvButtonLeft = contentView.findViewById(R.id.tv_button_left);
        tvButtonRight = contentView.findViewById(R.id.tv_button_right);
        viewLine = contentView.findViewById(R.id.view_line);

        //控件默认隐藏
        tvTitle.setVisibility(View.GONE);
        viewLine.setVisibility(View.GONE);
        ivIcon.setVisibility(View.GONE);
        tvMessage.setVisibility(View.GONE);
        tvButtonLeft.setVisibility(View.GONE);
        tvButtonRight.setVisibility(View.GONE);
        //构建Dialog
        setTitlText(p.title);
        setTitlTextSize(p.titleSizeSp);
        setImageResource(p.imageResource);
        setImageWidth(p.imageWidth);
        setImageHeight(p.imageHeight);
        setTvMessage(p.message1);
        setTvMessageGravity(p.message1Gravity);
        setCancelableFlag(p.isCanCancel);
        setLeftText(p.leftButtonText, p.leftListener);
        setLeftBtColor(p.leftBtColor);
        setRightText(p.rightButtontText, p.rightListener);
        setRightBtColor(p.rightBtColor);


    }
     /**
     * 设置标题
     *
     * @param title 标题文字
     */
    private void setTitlText(String title) {
        if (TextUtils.isEmpty(title)) {
            return;
        }
        tvTitle.setVisibility(View.VISIBLE);
        tvTitle.setText(title);
    }
    //......省略剩余控件代码
}

写完之后,我们来看看这个变形的建造者模式的Dialog是如何创建的:

DialogProduct.with(this)
        .title("提示")
        .message("您确认退出登录吗?")
        .canCancel(false)
        .leftBtColor(getResources().getColor(R.color.color_0090ff))
        .rightBtColor(getResources().getColor(R.color.color_f96c59))
        .leftBt("取消", new NormalDialog.ConcreteBuilder.ButtonClickLister() {
            @Override
            public void onClick(NormalDialog dialog) {
                dialog.cancel();
            }
        })
        .rightBt("确认", new NormalDialog.ConcreteBuilder.ButtonClickLister() {
            @Override
            public void onClick(NormalDialog dialog) {
                Toast.makeText(BuilderActivity.this, "退出登录成功!", Toast.LENGTH_SHORT).show();
                dialog.cancel();
            }
        })
        .create()
        .show();

总结

DialogProduct的代码我已经上传到了GitHub,各位看官可以自行食用。
这个Dialog现在也是项目A中的全局Dialog,使用起来也非常方便。
这里有几个细节可以和各位看官分享一下:
第一个就是按钮的点击事件设置,我将其与按钮文字内容的设置绑定在一起。因为我认为你设置了按钮,怎么可能会没有点击事件?
第二个就是按钮中间的分割线,是与右边按钮绑定的, 所以当只有一个按钮时,我们应该使用左边的按钮leftBt而不是右边的。
基本上就是这样啦!
希望这个Dialog可以给大家带来一些灵感。

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,858评论 25 707
  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 6,375评论 0 17
  • 面向对象的六大原则 单一职责原则 所谓职责是指类变化的原因。如果一个类有多于一个的动机被改变,那么这个类就具有多于...
    JxMY阅读 931评论 1 3
  • 长春,这座当时我为了爱情的圆满倾尽全力想方设法不顾家人的劝说要来的城市; 长春,这座后来我为了家庭...
    小鸭梨co阅读 386评论 0 0
  • 前几天坐电梯时偶然看到新换的广告,现在回想起来,脑子里只留下了智能、乐享几个词(估计属于李叫兽所谓的X型文案吧),...
    李侃理阅读 452评论 0 2