最近开发中遇到一个问题,就是在一个Activity启动后,后续的大量流程都由Dialog来承载。Android提供的Dialog其实大部分情况下只是一个Alert的作用,并不是一个能承载复杂业务的Controller,所以在Dialog之间互相跳转需要传递值或回调时,就会遇到问题。前人写的代码中包含了大量dialog互相传递的代码,比如ADialog跳转到BDialog之后,BDialog的某个操作需要调用A中的某个方法或传递值给A,于是出现了把A的引用传给B,把A中的Handler传给B这样的“没有办法的办法”的代码,看了甚是痛心。由于没有统一规范,当业务膨胀之后各种dialog互相传递导致各种耦合带来大量的问题,于是苦思冥想该如何解决这个问题。最近灵机一动,安卓开发者最熟悉的莫过于使用Activity了,如果能像使用Activity一样使用Dialog,那学习成本是不是就最低呢,于是乎,就冒出了写这么个框架的想法。
声明一下,这个框架目前只是个demo,也算是抛砖引玉,当然我们也可以借用EventBus这样的成熟框架,不过既然想到了,就当练手吧,希望各路大神如果看中这个思路希望一起来优化这个框架。
另外这个框架封装了基础的Dialog以及对Dialog展示消失制作自己的动画,前提是你有nineoldandroids的jar,好了,废话不多说,上代码。
package com.amuro.dialog_framework.base;
import com.amuro.dialog_framework.animation.BaseAnimatorSet;
import com.amuro.dialog_framework.utils.DialogIntent;
import com.amuro.dialog_framework.utils.DisplayUtils;
import com.nineoldandroids.animation.Animator;
import android.annotation.SuppressLint;
import android.app.Dialog;
import android.content.Context;
import android.graphics.Color;
import android.graphics.drawable.ColorDrawable;
import android.os.Bundle;
import android.view.Gravity;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnTouchListener;
import android.view.ViewGroup;
import android.view.Window;
import android.view.WindowManager.LayoutParams;
import android.widget.LinearLayout;
public abstract class BaseDialog extends Dialog
{
protected Context context;
private DialogIntent intent;
private int screenWidth;
private int screenHeight;
private int contentWidth;
private int contentHeight;
private int dialogMaxHeight;
private float widthScale;
private float heightScale;
private LinearLayout linearLayoutRoot;
private LinearLayout linearLayoutContent;
private BaseAnimatorSet showAnimation;
private BaseAnimatorSet dismissAnimation;
private boolean isShowAnimPlaying = false;
private boolean isDimissAnimPlaying = false;
private boolean isOutsideTouchDismiss = true;
private DialogManager dialogManager;
private int resultCode = -1;
private Bundle resultData;
public BaseDialog(Context context)
{
super(context);
this.context = context;
dialogManager = DialogManager.getInstance();
initData();
setDialogTheme();
initParams();
}
public DialogIntent getIntent()
{
return intent;
}
public void setIntent(DialogIntent intent)
{
this.intent = intent;
}
protected abstract void initParams();
private void initData()
{
screenWidth = DisplayUtils.getScreenWidth(context);
screenHeight = DisplayUtils.getScreenHeight(context);
dialogMaxHeight = screenHeight - DisplayUtils.getStatusBar(context);
widthScale = 0.88f;
heightScale = 0.5f;
}
private void setDialogTheme() {
requestWindowFeature(Window.FEATURE_NO_TITLE);// android:windowNoTitle
getWindow().setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));// android:windowBackground
getWindow().addFlags(LayoutParams.FLAG_DIM_BEHIND);// android:backgroundDimEnabled默认是true的
}
@Override
protected void onCreate(Bundle savedInstanceState)
{
linearLayoutRoot = new LinearLayout(context);
linearLayoutRoot.setGravity(Gravity.CENTER);
linearLayoutContent = new LinearLayout(context);
linearLayoutContent.setOrientation(LinearLayout.VERTICAL);
linearLayoutContent.setBackgroundColor(Color.WHITE);
linearLayoutContent.addView(onCreateContentView());
linearLayoutRoot.addView(linearLayoutContent);
setContentView(linearLayoutRoot,
new ViewGroup.LayoutParams(screenWidth, dialogMaxHeight));
}
protected abstract View onCreateContentView();
@Override
public void onAttachedToWindow()
{
super.onAttachedToWindow();
initContentView();
initShowAnimation();
}
private void initContentView()
{
if (widthScale == 0)
{
contentWidth = ViewGroup.LayoutParams.WRAP_CONTENT;
}
else if(widthScale == 1)
{
contentWidth = ViewGroup.LayoutParams.MATCH_PARENT;
}
else
{
contentWidth = (int) (screenWidth * widthScale);
}
if (heightScale == 0)
{
contentHeight = ViewGroup.LayoutParams.WRAP_CONTENT;
}
else if (heightScale == 1)
{
contentHeight = ViewGroup.LayoutParams.MATCH_PARENT;
}
else
{
contentHeight = (int) (dialogMaxHeight * heightScale);
}
linearLayoutContent.setLayoutParams(new LinearLayout.LayoutParams(contentWidth, contentHeight));
linearLayoutRoot.setOnTouchListener(new OnTouchListener()
{
@SuppressLint("ClickableViewAccessibility")
@Override
public boolean onTouch(View v, MotionEvent event)
{
if(isOutsideTouchDismiss)
{
int action = event.getAction();
if (action == MotionEvent.ACTION_DOWN)
{
if (!inRangeOfView(linearLayoutContent, event))
{
dismiss();
}
}
}
return false;
}
});
}
private void initShowAnimation()
{
if (showAnimation != null)
{
showAnimation.listener(new BaseAnimatorSet.AnimatorListener()
{
@Override
public void onAnimationStart(Animator animator)
{
isShowAnimPlaying = true;
}
@Override
public void onAnimationRepeat(Animator animator)
{
}
@Override
public void onAnimationEnd(Animator animator)
{
isShowAnimPlaying = false;
}
@Override
public void onAnimationCancel(Animator animator)
{
isShowAnimPlaying = false;
}
}).playOn(linearLayoutContent);
}
else
{
BaseAnimatorSet.reset(linearLayoutContent);
}
}
private boolean inRangeOfView(View view, MotionEvent ev)
{
int[] location = new int[2];
view.getLocationOnScreen(location);
int x = location[0];
int y = location[1];
if (ev.getX() < x || ev.getX() > (x + view.getWidth()) || ev.getY() < y
|| ev.getY() > (y + view.getHeight()))
{
return false;
}
return true;
}
@Override
public void dismiss()
{
if (dismissAnimation != null)
{
dismissAnimation.listener(new BaseAnimatorSet.AnimatorListener()
{
@Override
public void onAnimationStart(Animator animator)
{
isDimissAnimPlaying = true;
}
@Override
public void onAnimationRepeat(Animator animator)
{
}
@Override
public void onAnimationEnd(Animator animator)
{
isDimissAnimPlaying = false;
superDismiss();
}
@Override
public void onAnimationCancel(Animator animator)
{
isDimissAnimPlaying = false;
superDismiss();
}
}).playOn(linearLayoutContent);
}
else
{
superDismiss();
}
}
private void superDismiss()
{
super.dismiss();
}
@Override
public void onBackPressed()
{
if (isShowAnimPlaying || isDimissAnimPlaying)
{
return;
}
super.onBackPressed();
}
public BaseDialog setOutsideTouchDismiss(boolean isOutsideTouchDismiss)
{
this.isOutsideTouchDismiss = isOutsideTouchDismiss;
return this;
}
/** set window dim or not(设置背景是否昏暗) */
public BaseDialog setDimEnabled(boolean isDimEnabled)
{
if (isDimEnabled)
{
getWindow().addFlags(LayoutParams.FLAG_DIM_BEHIND);
}
else
{
getWindow().clearFlags(LayoutParams.FLAG_DIM_BEHIND);
}
return this;
}
/** set dialog width scale:0-1(设置对话框宽度,占屏幕宽的比例0-1) */
public BaseDialog setWidthScale(float widthScale)
{
this.widthScale = widthScale;
return this;
}
/** set dialog height scale:0-1(设置对话框高度,占屏幕宽的比例0-1) */
public BaseDialog setHeightScale(float heightScale)
{
this.heightScale = heightScale;
return this;
}
/** set show anim(设置显示的动画) */
public BaseDialog setShowAnim(BaseAnimatorSet showAnim)
{
this.showAnimation = showAnim;
return this;
}
/** set dismiss anim(设置隐藏的动画) */
public BaseDialog setDismissAnim(BaseAnimatorSet dismissAnim)
{
this.dismissAnimation = dismissAnim;
return this;
}
protected void setResult(int resultCode)
{
this.resultCode = resultCode;
this.resultData = null;
}
protected void setResult(int resultCode, Bundle data)
{
this.resultCode = resultCode;
this.resultData = data;
}
public int getResultCode()
{
return resultCode;
}
public Bundle getResultData()
{
return resultData;
}
protected void startDialogForResult(DialogIntent intent, int requestCode)
{
dialogManager.startDialogForResult(intent, requestCode);
}
protected void onDialogResult(int requestCode, int resultCode, Bundle data)
{
}
}
BaseDialog这种也算是标准配置了,解释一下思路:
1. 安卓的Window和View的关系我就不多说了,不理解的去看读书笔记二。我们按照标准的Dialog配置,把Window的背景置为透明,加上阴影,去掉丑陋的系统Title。然后丢一个Root LinearLayout在后面,它的大小就是全屏,然后背景也是透明。
2. Root上面再加一个content LinearLayout作为承载我们所有自定义View的容器,而这个content的大小通过widthScale和heightScale来调整,默认就是系统AlertDialog差不多的大小。
3. 因为背景已经完全被root覆盖,所以需要我们自己处理onTouchOutsideDismiss方法,思路很简单,root被点击的时候判断这个点的坐标是否在content范围内,是就屏蔽事件,不是就dismiss。
4. 重点就是startDialogForResult方法和onDialogResult方法,是不是很眼熟,没错,就是让你感觉和Activity一样。逻辑也是差不多的,startDialogForResult方法调用了DialogManager的startDialogForResult方法,onDialogResult方法则为空方法,待子类自己去玩。
5. 动画的基类为BaseAnimatorSet,封装了必要的动画方法,提供接口供调用者自行设置想要的动画。
下面先来看DialogManager类:
package com.amuro.dialog_framework.base;
import java.lang.reflect.Constructor;
import android.content.Context;
import android.content.DialogInterface;
import android.content.DialogInterface.OnDismissListener;
import com.amuro.dialog_framework.utils.DialogIntent;
public class DialogManager
{
private DialogManager(){}
private final static class DialogManagerHolder
{
private static DialogManager instance = new DialogManager();
}
public static DialogManager getInstance()
{
return DialogManagerHolder.instance;
}
private Context context;
private BaseDialog fromDialog;
private BaseDialog toDialog;
private int requestCode;
public DialogManager setContext(Context context)
{
if(this.context == null)
{
this.context = context;
}
return this;
}
public void startDialog(DialogIntent intent)
{
startDialogForResult(intent, -1);
}
public void startDialogForResult(DialogIntent intent, int requestCode)
{
try
{
fromDialog = intent.getFromDialog();
Constructor<? extends BaseDialog> c =
intent.getDialogClass().getConstructor(Context.class);
this.requestCode = requestCode;
toDialog = c.newInstance(context);
toDialog.setIntent(intent);
toDialog.setOnDismissListener(new OnDismissListener()
{
@Override
public void onDismiss(DialogInterface dialog)
{
notifyResult();
}
});
toDialog.show();
}
catch (Exception e)
{
e.printStackTrace();
}
}
private void notifyResult()
{
if(fromDialog == null)
{
return;
}
if(requestCode == -1)
{
return;
}
fromDialog.onDialogResult(requestCode, toDialog.getResultCode(), toDialog.getResultData());
}
}
一看就懂的代码,记录发出请求的Dialog,再反射拿到新的Dialog,在新的Dialog dismiss的时候,回调请求Dialog的onDialogResult方法,其形式和Activity几乎如出一辙。当然demo就是demo,这里只是最简单的实现,后续可以通过参考Activity stack和map来维护整个dialog队列,实现更复杂的需求。为了传递数据方便,这里干脆封装了Intent,弄了个DialogIntent,复杂的数据传递,就可以直接复用Intent的代码啦~
package com.amuro.dialog_framework.utils;
import com.amuro.dialog_framework.base.BaseDialog;
import android.content.Intent;
public class DialogIntent extends Intent
{
private BaseDialog fromDialog;
private Class<? extends BaseDialog> toDialogClass;
public DialogIntent(BaseDialog fromDialog,
Class<? extends BaseDialog> dialogClass)
{
super();
this.fromDialog = fromDialog;
this.toDialogClass = dialogClass;
}
public BaseDialog getFromDialog()
{
return fromDialog;
}
public void setFromDialog(BaseDialog fromDialog)
{
this.fromDialog = fromDialog;
}
public Class<? extends BaseDialog> getDialogClass()
{
return toDialogClass;
}
public void setDialogClass(Class<? extends BaseDialog> dialogClass)
{
this.toDialogClass = dialogClass;
}
}
好了,最后再看下动画的基类BaseAnimatorSet:、
package com.amuro.dialog_framework.animation;
import android.view.View;
import android.view.animation.Interpolator;
import com.nineoldandroids.animation.Animator;
import com.nineoldandroids.animation.AnimatorSet;
import com.nineoldandroids.view.ViewHelper;
public abstract class BaseAnimatorSet
{
public interface AnimatorListener
{
void onAnimationStart(Animator animator);
void onAnimationRepeat(Animator animator);
void onAnimationEnd(Animator animator);
void onAnimationCancel(Animator animator);
}
protected long duration = 500;
protected AnimatorSet animatorSet = new AnimatorSet();
private Interpolator interpolator;
private long delay;
private AnimatorListener listener;
public abstract void setAnimation(View view);
protected void start(final View view)
{
reset(view);
setAnimation(view);
animatorSet.setDuration(duration);
if (interpolator != null)
{
animatorSet.setInterpolator(interpolator);
}
if (delay > 0)
{
animatorSet.setStartDelay(delay);
}
if (listener != null)
{
animatorSet.addListener(new Animator.AnimatorListener()
{
@Override
public void onAnimationStart(Animator animator)
{
listener.onAnimationStart(animator);
}
@Override
public void onAnimationRepeat(Animator animator)
{
listener.onAnimationRepeat(animator);
}
@Override
public void onAnimationEnd(Animator animator)
{
listener.onAnimationEnd(animator);
}
@Override
public void onAnimationCancel(Animator animator)
{
listener.onAnimationCancel(animator);
}
});
}
animatorSet.start();
}
/** 设置动画时长 */
public BaseAnimatorSet duration(long duration)
{
this.duration = duration;
return this;
}
/** 设置动画时长 */
public BaseAnimatorSet delay(long delay)
{
this.delay = delay;
return this;
}
/** 设置动画插补器 */
public BaseAnimatorSet interpolator(Interpolator interpolator)
{
this.interpolator = interpolator;
return this;
}
/** 动画监听 */
public BaseAnimatorSet listener(AnimatorListener listener)
{
this.listener = listener;
return this;
}
/** 在View上执行动画 */
public void playOn(View view)
{
start(view);
}
public static void reset(View view)
{
ViewHelper.setAlpha(view, 1);
ViewHelper.setScaleX(view, 1);
ViewHelper.setScaleY(view, 1);
ViewHelper.setTranslationX(view, 0);
ViewHelper.setTranslationY(view, 0);
ViewHelper.setRotation(view, 0);
ViewHelper.setRotationY(view, 0);
ViewHelper.setRotationX(view, 0);
}
}
关于属性动画可参考相关的blog,这里不再赘述,想要实现自己酷炫的动画效果,就继承之后折腾吧。下面贴三个实现类来展示一下使用效果,Main,Login,Register
package com.amuro.example.dialogs;
import android.content.Context;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import com.amuro.dialog_framework.animation.AnimatorBottomExit;
import com.amuro.dialog_framework.animation.AnimatorTopEnter;
import com.amuro.dialog_framework.animation.BaseAnimatorSet;
import com.amuro.dialog_framework.base.BaseAnimationDialog;
import com.amuro.dialog_framework.utils.DialogIntent;
import com.amuro.dialogframework.R;
import com.amuro.example.bean.User;
import com.amuro.example.utils.custom_view.TitleBar;
import com.amuro.example.utils.custom_view.ToastUtils;
public class MainDialog extends BaseAnimationDialog
{
private TitleBar titleBar;
private Button button1;
private Button button2;
public MainDialog(Context context)
{
super(context);
}
@Override
protected BaseAnimatorSet getShowAnim()
{
return new AnimatorTopEnter();
}
@Override
protected BaseAnimatorSet getDismissAnim()
{
return new AnimatorBottomExit();
}
@Override
protected View onCreateContentView()
{
View view = View.inflate(context, R.layout.dialog_main_layout, null);
return view;
}
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
initTitle();
initView();
}
private void initTitle()
{
titleBar = (TitleBar)findViewById(R.id.tb);
titleBar.setTitleText("主页");
}
private void initView()
{
button1 = (Button)findViewById(R.id.bt1);
button1.setOnClickListener(new View.OnClickListener()
{
@Override
public void onClick(View v)
{
DialogIntent intent = new DialogIntent(MainDialog.this, LoginDialog.class);
intent.putExtra("test", "From MainDialog");
startDialogForResult(intent, 1);
dismiss();
}
});
button2 = (Button)findViewById(R.id.bt2);
button2.setOnClickListener(new View.OnClickListener()
{
@Override
public void onClick(View v)
{
DialogIntent intent = new DialogIntent(MainDialog.this, RegisterDialog.class);
intent.putExtra("test", "From MainDialog");
startDialogForResult(intent, 2);
}
});
}
@Override
protected void onDialogResult(int requestCode, int resultCode, Bundle data)
{
if(requestCode == 1)
{
if(resultCode == LoginDialog.LOGIN_SUCCESS)
{
User user = null;
if(data != null)
{
user = data.getParcelable("User");
}
String userName = "null";
if(user != null)
{
userName = user.getName();
}
ToastUtils.showToast(context, "登录成功, 登录用户为:" + userName);
}
else if(resultCode == LoginDialog.LOGIN_FAILED)
{
ToastUtils.showToast(context, "登录失败");
}
else if(resultCode == LoginDialog.LOGIN_CANCEL)
{
ToastUtils.showToast(context, "登录取消");
}
}
else if(requestCode == 2)
{
if(resultCode == RegisterDialog.REGISTER_SUCCESS)
{
User user = null;
if(data != null)
{
user = data.getParcelable("User");
}
String userName = "null";
if(user != null)
{
userName = user.getName();
}
ToastUtils.showToast(context, "注册成功, 注册用户为:" + userName);
}
else if(resultCode == RegisterDialog.REGISTER_FAILED)
{
ToastUtils.showToast(context, "注册失败");
}
else if(resultCode == RegisterDialog.REGISTER_CANCEL)
{
ToastUtils.showToast(context, "注册取消");
}
}
}
}
package com.amuro.example.dialogs;
import android.content.Context;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import com.amuro.dialog_framework.animation.AnimatorBottomExit;
import com.amuro.dialog_framework.animation.AnimatorTopEnter;
import com.amuro.dialog_framework.animation.BaseAnimatorSet;
import com.amuro.dialog_framework.base.BaseAnimationDialog;
import com.amuro.dialogframework.R;
import com.amuro.example.bean.User;
import com.amuro.example.utils.custom_view.TitleBar;
import com.amuro.example.utils.custom_view.ToastUtils;
public class LoginDialog extends BaseAnimationDialog
{
public static final int LOGIN_SUCCESS = 0;
public static final int LOGIN_FAILED = 1;
public static final int LOGIN_CANCEL = 2;
private TitleBar titleBar;
private Button button1;
private Button button2;
public LoginDialog(Context context)
{
super(context);
}
@Override
protected BaseAnimatorSet getShowAnim()
{
return new AnimatorTopEnter();
}
@Override
protected BaseAnimatorSet getDismissAnim()
{
return new AnimatorBottomExit();
}
@Override
protected void initParams()
{
super.initParams();
setOutsideTouchDismiss(false);
}
@Override
protected View onCreateContentView()
{
return View.inflate(context, R.layout.dialog_login_layout, null);
}
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
String test = getIntent().getStringExtra("test");
ToastUtils.showToast(context, "来自上一个Dialog的msg: " + test);
initTitle();
initView();
}
private void initTitle()
{
titleBar = (TitleBar)findViewById(R.id.tb);
titleBar.setTitleText("登录");
}
private void initView()
{
button1 = (Button)findViewById(R.id.bt_success);
button1.setOnClickListener(new View.OnClickListener()
{
@Override
public void onClick(View v)
{
Bundle bundle = new Bundle();
User user = new User("1", "张三");
bundle.putParcelable("User", user);
setResult(LOGIN_SUCCESS, bundle);
dismiss();
}
});
button2 = (Button)findViewById(R.id.bt_fail);
button2.setOnClickListener(new View.OnClickListener()
{
@Override
public void onClick(View v)
{
setResult(LOGIN_FAILED);
dismiss();
}
});
}
@Override
public void onBackPressed()
{
super.onBackPressed();
setResult(LOGIN_CANCEL);
}
}
package com.amuro.example.dialogs;
import android.content.Context;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import com.amuro.dialog_framework.base.BaseDialog;
import com.amuro.dialogframework.R;
import com.amuro.example.bean.User;
import com.amuro.example.utils.custom_view.TitleBar;
import com.amuro.example.utils.custom_view.ToastUtils;
public class RegisterDialog extends BaseDialog
{
public static final int REGISTER_SUCCESS = 0;
public static final int REGISTER_FAILED = 1;
public static final int REGISTER_CANCEL = 2;
private TitleBar titleBar;
private Button button1;
private Button button2;
public RegisterDialog(Context context)
{
super(context);
}
@Override
protected void initParams()
{
setOutsideTouchDismiss(false);
}
@Override
protected View onCreateContentView()
{
return View.inflate(context, R.layout.dialog_register_layout, null);
}
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
String test = getIntent().getStringExtra("test");
ToastUtils.showToast(context, "来自上一个Dialog的msg: " + test);
initTitle();
initView();
}
private void initTitle()
{
titleBar = (TitleBar)findViewById(R.id.tb);
titleBar.setTitleText("注册");
}
private void initView()
{
button1 = (Button)findViewById(R.id.bt_success);
button1.setOnClickListener(new View.OnClickListener()
{
@Override
public void onClick(View v)
{
Bundle bundle = new Bundle();
User user = new User("1", "李四");
bundle.putParcelable("User", user);
setResult(REGISTER_SUCCESS, bundle);
dismiss();
}
});
button2 = (Button)findViewById(R.id.bt_fail);
button2.setOnClickListener(new View.OnClickListener()
{
@Override
public void onClick(View v)
{
setResult(REGISTER_FAILED);
dismiss();
}
});
}
@Override
public void onBackPressed()
{
super.onBackPressed();
setResult(REGISTER_CANCEL);
}
}
好了,Dialog之间通信再也不必用蛋疼的监听器和互相传引用传handler了,像Activity一样,你值得拥有。