设计模式之建造模式(建造模式实现消息提示DialogFragment)

一、什么是建造模式

建造者模式:是将一个复杂的对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。

举一个栗子,我有A、B、C、D、E、F六块积木。我拿过来一个机器,我只需要向机器中放入积木,机器就可以根据我放入的积木生成不同的模型给我。此中,模型为“复杂的对象”,积木为“表现”,机器的作用是“构建对象”。当我构建某个对象时,我只需要知道它有什么样的表现,而不需要管它是如何构建的。

建造者模式通常包括下面几个角色:

1、Builder:给出一个抽象接口,以规范产品对象的各个组成成分的建造。这个接口规定要实现复杂对象的哪些部分的创建,并不涉及具体的对象部件的创建。
2、ConcreteBuilder:实现Builder接口,针对不同的商业逻辑,具体化复杂对象的各部分的创建。 在建造过程完成后,提供产品的实例。
3、Director:调用具体建造者来创建复杂对象的各个部分,在指导者中不涉及具体产品的信息,只负责保证对象各部分完整创建或按某种顺序创建。
4、Product:要创建的复杂对象。

二、演示实例(Android中使用DialogFragment构建消息提示框)

在Android应用程序中需要给予用户不同的消息提示框。如:仅有一个ok按钮的信息提示框,含有取消和确认的消息选择框,有标题有内容的消息选择框等等。
提示框上所有的元素可以视为它所包含的“表现”,最后展示给用户的提示框为“复杂的对象”。我们需要做的就是封装一个构建最终展示给用户的提示框的类。

代码制造中与定义不符之处

忽略了Director角色类的定义,Director角色的作用是将“复杂对象”的要组合的“表现”设置到ConcreteBuilder角色中。而ConcreteBuilder角色的作用是将接收到的“表现”进行建造组合后,提供产品实例。

二、演示实例

实际代码内容

代码制造内容中使用了DataBing,用DataBing完成Fragment与布局文件间数据的双向绑定。

1、Product:要创建的复杂对象。
代码块
public class PromptDialogFragment extends DialogFragment {

    private final static String TAG = "PromptDialogFragment";

    private static final String BUNDLE_KEY_INFO = "BUNDLE_KEY_INFO";
    private static final String BUNDLE_KEY_CANCEL = "BUNDLE_KEY_CANCEL";
    private static final String BUNDLE_KEY_CONFIRM = "BUNDLE_KEY_CONFIRM";
    private static final String BUNDLE_KEY_OK = "BUNDLE_KEY_OK";
    private static final String BUNDLE_KEY_CALLBACK = "BUNDLE_KEY_CALLBACK";

    /**
     * Dialog提示框的默认宽度
     */
    private final int DEFAULT_WIDTH = 240;

    @Inject
    DisplayUtil displayUtil;

    FragmentPromptDialogBinding binding;

    public final ObservableField<String> infoValue = new ObservableField<>();
    public final ObservableField<String> cancelValue = new ObservableField<>();
    public final ObservableField<String> confirmValue = new ObservableField<>();
    public final ObservableField<String> okValue = new ObservableField<>();

    private CallBack callBack;

    public static PromptDialogFragment newInstance(){
        PromptDialogFragment fragment = new PromptDialogFragment();
        return fragment;
    }

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        AndroidSupportInjection.inject(this);
        Bundle bundle = getArguments();
        infoValue.set(bundle.getString(BUNDLE_KEY_INFO));
        cancelValue.set(bundle.getString(BUNDLE_KEY_CANCEL));
        confirmValue.set(bundle.getString(BUNDLE_KEY_CONFIRM));
        okValue.set(bundle.getString(BUNDLE_KEY_OK));
        callBack= (CallBack) bundle.getSerializable(BUNDLE_KEY_CALLBACK);
    }

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        //设置DialogFragment无标题
        getDialog().requestWindowFeature(Window.FEATURE_NO_TITLE);
        binding = DataBindingUtil.inflate(inflater, R.layout.fragment_prompt_dialog, container, false);
        binding.setViewModel(this);
        //设置点击空白区域Dialog框不消失
        setCancelable(false);

        return binding.getRoot();
    }

    @Override
    public void onStart() {
        super.onStart();
        initDialogSize();
    }

    /**
     * 初始化Dialog提示框的大小
     */
    private void initDialogSize(){
        Window window = getDialog().getWindow();
        // 一定要设置Background,如果不设置,window属性设置无效
        window.setBackgroundDrawable( new ColorDrawable(Color.TRANSPARENT));

        DisplayMetrics dm = new DisplayMetrics();
        getActivity().getWindowManager().getDefaultDisplay().getMetrics( dm );

        WindowManager.LayoutParams params = window.getAttributes();
        params.gravity = Gravity.CENTER;
        // 使用ViewGroup.LayoutParams,以便Dialog 宽度充满整个屏幕
        params.width =  displayUtil.dip2px(DEFAULT_WIDTH);
        params.height = ViewGroup.LayoutParams.WRAP_CONTENT;
        window.setAttributes(params);
    }

    public void show(FragmentManager manager) {
        super.show(manager, TAG);
    }

    /**
     * 消息框取消按钮的点击事件(左边按钮)
     * @param view
     */
    public void cancelClick(View view){
        super.dismiss();
        if(callBack!=null){
            callBack.onCancel();
        }
    }

    /**
     * 消息框确认按钮的点击事件(右边按钮)
     * @param view
     */
    public void confirmClick(View view){
        super.dismiss();
        if(callBack!=null){
            callBack.onSuccess();
        }
    }

    /**
     * 消息框OK按钮的点击事件(仅有一个按钮)
     * @param view
     */
    public void okClick(View view){
        super.dismiss();
        if(callBack!=null){
            callBack.onSuccess();
        }
    }
    /**
     * Dialog提示框按钮点击事件回调接口
     */
    public interface CallBack extends Serializable {
        void onSuccess();
        void onCancel();
    }
}
2、Builder:给出一个抽象接口,以规范产品对象的各个组成成分的建造。这个接口规定要实现复杂对象的哪些部分的创建,并不涉及具体的对象部件的创建。
代码块
   /**
    * 抽象建造,以规范产品对象的各个组成成分的建造
    */
    public interface PromptDialogBuilder{
        /**
         * 设置提示框提示信息
         * @param value
         * @return
         */
        PromptDialogBuilder setInfoValue(String value);

        /**
         * 设置取消按钮文言(左侧按钮)
         * @param value
         * @return
         */
        PromptDialogBuilder setCancelValue(String value);

        /**
         * 设置确认按钮文言(右侧按钮)
         * @param value
         * @return
         */
        PromptDialogBuilder setConfirmValue(String value);

        /**
         * 设置按钮文言(该按钮与“取消、确认按钮在显示上互斥”)
         * @param value
         * @return
         */
        PromptDialogBuilder setOKValue(String value);

        /**
         * 设置按钮点击事件的回调
         * @param callBack
         * @return
         */
        PromptDialogBuilder setCallBack(CallBack callBack);

        /**
         * 提供产品实例(返回PromptDialogFragment对象)
         * @return
         */
        PromptDialogFragment build();
    }
3、ConcreteBuilder:实现Builder接口,针对不同的商业逻辑,具体化复杂对象的各部分的创建。 在建造过程完成后,提供产品的实例。
代码块
    /**
     * 具体的建造,针对不同的商业逻辑,具体化复杂对象的各部分的创建。 在建造过程完成后,提供产品的实例。
     */
    public final static class Builder implements PromptDialogBuilder{

        private String infoValue;
        private String cancelValue = "取消";
        private String confirmValue = "确认";
        private String okValue;
        private CallBack callBack;

        private AppCompatActivity activity;

        public <T extends AppCompatActivity>Builder(T activity){
            this.activity=activity;
        }

        @Override
        public Builder setInfoValue(String value) {
            this.infoValue=value;
            return this;
        }

        @Override
        public Builder setCancelValue(String value) {
            this.cancelValue=value;
            return this;
        }

        @Override
        public Builder setConfirmValue(String value) {
            this.confirmValue=value;
            return this;
        }

        @Override
        public Builder setOKValue(String value) {
            this.okValue=value;
            return this;
        }

        @Override
        public Builder setCallBack(CallBack callBack) {
            this.callBack=callBack;
            return this;
        }

        @Override
        public PromptDialogFragment build() {
            PromptDialogFragment fragment = PromptDialogFragment.newInstance();
            Bundle bundle=new Bundle();
            bundle.putString(BUNDLE_KEY_INFO, infoValue);
            bundle.putString(BUNDLE_KEY_CANCEL, cancelValue);
            bundle.putString(BUNDLE_KEY_CONFIRM, confirmValue);
            bundle.putString(BUNDLE_KEY_OK, okValue);
            bundle.putSerializable(BUNDLE_KEY_CALLBACK, callBack);
            fragment.setArguments(bundle);
            fragment.show(activity.getSupportFragmentManager());
            return fragment;
        }

    }
4、Director:调用具体建造者来创建复杂对象的各个部分,在指导者中不涉及具体产品的信息,只负责保证对象各部分完整创建或按某种顺序创建。
代码块
new PromptDialogFragment.Builder(this).setInfoValue("您已登录成功!").setOKValue("OK").build();
代码块
new PromptDialogFragment.Builder(this)
                .setInfoValue("你是否执行登录操作!")
                .setCancelValue("取消")
                .setConfirmValue("确认")
                .setCallBack(new PromptDialogFragment.CallBack() {
                    @Override
                    public void onSuccess() {
                        Toast.makeText(activity, "确认操作", Toast.LENGTH_LONG).show();
                    }

                    @Override
                    public void onCancel() {
                        Toast.makeText(activity, "取消操作", Toast.LENGTH_LONG).show();
                    }
                }).build();

最后附上布局文件的内容

代码块
<layout xmlns:android="http://schemas.android.com/apk/res/android">

    <data>
        <import type="android.view.View"/>
        <variable
            name="viewModel"
            type="com..fragment.PromptDialogFragment" />

    </data>

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="@drawable/white_round_bg"
        android:paddingTop="20dp">

        <TextView
            android:id="@+id/prompt_info"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginLeft="15dp"
            android:layout_marginRight="15dp"
            android:gravity="center_horizontal"
            android:text="@{viewModel.infoValue}" />

        <View
            android:id="@+id/prompt_lines"
            android:layout_width="match_parent"
            android:layout_height="0.1dp"
            android:layout_below="@+id/prompt_info"
            android:layout_marginTop="15dp"
            android:background="@color/new_title_line_bg" />

        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="40dp"
            android:layout_below="@+id/prompt_lines"
            android:visibility="@{viewModel.okValue==null?View.VISIBLE:View.GONE}">

            <TextView
                android:layout_width="0dp"
                android:layout_height="match_parent"
                android:layout_weight="1"
                android:gravity="center"
                android:text="@{viewModel.cancelValue}"
                android:onClick="@{viewModel::cancelClick}"/>

            <View
                android:layout_width="0.1dp"
                android:layout_height="match_parent"
                android:layout_below="@+id/prompt_info"
                android:layout_marginBottom="3dp"
                android:layout_marginTop="3dp"
                android:background="@color/new_title_line_bg" />

            <TextView
                android:layout_width="0dp"
                android:layout_height="match_parent"
                android:layout_weight="1"
                android:gravity="center"
                android:text="@{viewModel.confirmValue}"
                android:onClick="@{viewModel::confirmClick}"/>
        </LinearLayout>

        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="40dp"
            android:layout_below="@+id/prompt_lines"
            android:visibility="@{viewModel.okValue==null?View.GONE:View.VISIBLE}">

            <TextView
                android:layout_width="0dp"
                android:layout_height="match_parent"
                android:layout_weight="1"
                android:gravity="center"
                android:text="@{viewModel.okValue}"
                android:onClick="@{viewModel::okClick}"/>
        </LinearLayout>


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

推荐阅读更多精彩内容