DialogInit.java 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430
  1. package com.afollestad.materialdialogs;
  2. import android.content.Context;
  3. import android.content.res.ColorStateList;
  4. import android.content.res.Resources;
  5. import android.graphics.drawable.Drawable;
  6. import android.graphics.drawable.GradientDrawable;
  7. import android.os.Build;
  8. import android.support.annotation.LayoutRes;
  9. import android.support.annotation.NonNull;
  10. import android.support.annotation.StyleRes;
  11. import android.support.annotation.UiThread;
  12. import android.text.InputType;
  13. import android.text.method.LinkMovementMethod;
  14. import android.text.method.PasswordTransformationMethod;
  15. import android.view.View;
  16. import android.view.ViewGroup;
  17. import android.widget.EditText;
  18. import android.widget.FrameLayout;
  19. import android.widget.ImageView;
  20. import android.widget.ListView;
  21. import android.widget.ProgressBar;
  22. import android.widget.ScrollView;
  23. import android.widget.TextView;
  24. import com.afollestad.materialdialogs.internal.MDButton;
  25. import com.afollestad.materialdialogs.internal.MDRootLayout;
  26. import com.afollestad.materialdialogs.internal.MDTintHelper;
  27. import com.afollestad.materialdialogs.progress.CircularProgressDrawable;
  28. import com.afollestad.materialdialogs.simplelist.MaterialSimpleListAdapter;
  29. import com.afollestad.materialdialogs.util.DialogUtils;
  30. import java.util.ArrayList;
  31. import java.util.Arrays;
  32. /**
  33. * Used by MaterialDialog while initializing the dialog. Offloads some of the code to make the main class
  34. * cleaner and easier to read/maintain.
  35. *
  36. * @author Aidan Follestad (afollestad)
  37. */
  38. class DialogInit {
  39. @StyleRes
  40. public static int getTheme(@NonNull MaterialDialog.Builder builder) {
  41. boolean darkTheme = DialogUtils.resolveBoolean(builder.context, R.attr.md_dark_theme, builder.theme == Theme.DARK);
  42. builder.theme = darkTheme ? Theme.DARK : Theme.LIGHT;
  43. return darkTheme ? R.style.MD_Dark : R.style.MD_Light;
  44. }
  45. @LayoutRes
  46. public static int getInflateLayout(MaterialDialog.Builder builder) {
  47. if (builder.customView != null) {
  48. return R.layout.md_dialog_custom;
  49. } else if (builder.items != null && builder.items.length > 0 || builder.adapter != null) {
  50. return R.layout.md_dialog_list;
  51. } else if (builder.progress > -2) {
  52. return R.layout.md_dialog_progress;
  53. } else if (builder.indeterminateProgress) {
  54. if (builder.indeterminateIsHorizontalProgress)
  55. return R.layout.md_dialog_progress_indeterminate_horizontal;
  56. return R.layout.md_dialog_progress_indeterminate;
  57. } else if (builder.inputCallback != null) {
  58. return R.layout.md_dialog_input;
  59. } else {
  60. return R.layout.md_dialog_basic;
  61. }
  62. }
  63. @UiThread
  64. public static void init(final MaterialDialog dialog) {
  65. final MaterialDialog.Builder builder = dialog.mBuilder;
  66. // Set cancelable flag and dialog background color
  67. dialog.setCancelable(builder.cancelable);
  68. dialog.setCanceledOnTouchOutside(builder.cancelable);
  69. if (builder.backgroundColor == 0)
  70. builder.backgroundColor = DialogUtils.resolveColor(builder.context, R.attr.md_background_color);
  71. if (builder.backgroundColor != 0) {
  72. GradientDrawable drawable = new GradientDrawable();
  73. drawable.setCornerRadius(builder.context.getResources().getDimension(R.dimen.md_bg_corner_radius));
  74. drawable.setColor(builder.backgroundColor);
  75. DialogUtils.setBackgroundCompat(dialog.view, drawable);
  76. }
  77. // Retrieve action button colors from theme attributes or the Builder
  78. if (!builder.positiveColorSet)
  79. builder.positiveColor = DialogUtils.resolveActionTextColorStateList(builder.context, R.attr.md_positive_color, builder.positiveColor);
  80. if (!builder.neutralColorSet)
  81. builder.neutralColor = DialogUtils.resolveActionTextColorStateList(builder.context, R.attr.md_neutral_color, builder.neutralColor);
  82. if (!builder.negativeColorSet)
  83. builder.negativeColor = DialogUtils.resolveActionTextColorStateList(builder.context, R.attr.md_negative_color, builder.negativeColor);
  84. if (!builder.widgetColorSet)
  85. builder.widgetColor = DialogUtils.resolveColor(builder.context, R.attr.md_widget_color, builder.widgetColor);
  86. // Retrieve default title/content colors
  87. if (!builder.titleColorSet) {
  88. final int titleColorFallback = DialogUtils.resolveColor(dialog.getContext(), android.R.attr.textColorPrimary);
  89. builder.titleColor = DialogUtils.resolveColor(builder.context, R.attr.md_title_color, titleColorFallback);
  90. // if (builder.titleColor == titleColorFallback) {
  91. // // Only check for light/dark if color wasn't set to md_title_color
  92. // if (DialogUtils.isColorDark(builder.titleColor)) {
  93. // if (builder.theme == Theme.DARK)
  94. // builder.titleColor = DialogUtils.resolveColor(builder.context, android.R.attr.textColorPrimaryInverse);
  95. // } else if (builder.theme == Theme.LIGHT)
  96. // builder.titleColor = DialogUtils.resolveColor(builder.context, android.R.attr.textColorPrimaryInverse);
  97. // }
  98. }
  99. if (!builder.contentColorSet) {
  100. final int contentColorFallback = DialogUtils.resolveColor(dialog.getContext(), android.R.attr.textColorSecondary);
  101. builder.contentColor = DialogUtils.resolveColor(builder.context, R.attr.md_content_color, contentColorFallback);
  102. // if (builder.contentColor == contentColorFallback) {
  103. // // Only check for light/dark if color wasn't set to md_content_color
  104. // if (DialogUtils.isColorDark(builder.contentColor)) {
  105. // if (builder.theme == Theme.DARK)
  106. // builder.contentColor = DialogUtils.resolveColor(builder.context, android.R.attr.textColorSecondaryInverse);
  107. // } else if (builder.theme == Theme.LIGHT)
  108. // builder.contentColor = DialogUtils.resolveColor(builder.context, android.R.attr.textColorSecondaryInverse);
  109. // }
  110. }
  111. if (!builder.itemColorSet)
  112. builder.itemColor = DialogUtils.resolveColor(builder.context, R.attr.md_item_color, builder.contentColor);
  113. // Retrieve references to views
  114. dialog.title = (TextView) dialog.view.findViewById(R.id.title);
  115. dialog.icon = (ImageView) dialog.view.findViewById(R.id.icon);
  116. dialog.titleFrame = dialog.view.findViewById(R.id.titleFrame);
  117. dialog.content = (TextView) dialog.view.findViewById(R.id.content);
  118. dialog.listView = (ListView) dialog.view.findViewById(R.id.contentListView);
  119. // Button views initially used by checkIfStackingNeeded()
  120. dialog.positiveButton = (MDButton) dialog.view.findViewById(R.id.buttonDefaultPositive);
  121. dialog.neutralButton = (MDButton) dialog.view.findViewById(R.id.buttonDefaultNeutral);
  122. dialog.negativeButton = (MDButton) dialog.view.findViewById(R.id.buttonDefaultNegative);
  123. // Don't allow the submit button to not be shown for input dialogs
  124. if (builder.inputCallback != null && builder.positiveText == null)
  125. builder.positiveText = builder.context.getText(android.R.string.ok);
  126. // Set up the initial visibility of action buttons based on whether or not text was set
  127. dialog.positiveButton.setVisibility(builder.positiveText != null ? View.VISIBLE : View.GONE);
  128. dialog.neutralButton.setVisibility(builder.neutralText != null ? View.VISIBLE : View.GONE);
  129. dialog.negativeButton.setVisibility(builder.negativeText != null ? View.VISIBLE : View.GONE);
  130. // Setup icon
  131. if (builder.icon != null) {
  132. dialog.icon.setVisibility(View.VISIBLE);
  133. dialog.icon.setImageDrawable(builder.icon);
  134. } else {
  135. Drawable d = DialogUtils.resolveDrawable(builder.context, R.attr.md_icon);
  136. if (d != null) {
  137. dialog.icon.setVisibility(View.VISIBLE);
  138. dialog.icon.setImageDrawable(d);
  139. } else {
  140. dialog.icon.setVisibility(View.GONE);
  141. }
  142. }
  143. // Setup icon size limiting
  144. int maxIconSize = builder.maxIconSize;
  145. if (maxIconSize == -1)
  146. maxIconSize = DialogUtils.resolveDimension(builder.context, R.attr.md_icon_max_size);
  147. if (builder.limitIconToDefaultSize || DialogUtils.resolveBoolean(builder.context, R.attr.md_icon_limit_icon_to_default_size))
  148. maxIconSize = builder.context.getResources().getDimensionPixelSize(R.dimen.md_icon_max_size);
  149. if (maxIconSize > -1) {
  150. dialog.icon.setAdjustViewBounds(true);
  151. dialog.icon.setMaxHeight(maxIconSize);
  152. dialog.icon.setMaxWidth(maxIconSize);
  153. dialog.icon.requestLayout();
  154. }
  155. // Setup divider color in case content scrolls
  156. if (!builder.dividerColorSet) {
  157. final int dividerFallback = DialogUtils.resolveColor(dialog.getContext(), R.attr.md_divider);
  158. builder.dividerColor = DialogUtils.resolveColor(builder.context, R.attr.md_divider_color, dividerFallback);
  159. }
  160. dialog.view.setDividerColor(builder.dividerColor);
  161. // Setup title and title frame
  162. if (dialog.title != null) {
  163. dialog.setTypeface(dialog.title, builder.mediumFont);
  164. dialog.title.setTextColor(builder.titleColor);
  165. dialog.title.setGravity(builder.titleGravity.getGravityInt());
  166. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
  167. //noinspection ResourceType
  168. dialog.title.setTextAlignment(builder.titleGravity.getTextAlignment());
  169. }
  170. if (builder.title == null) {
  171. dialog.titleFrame.setVisibility(View.GONE);
  172. } else {
  173. dialog.title.setText(builder.title);
  174. dialog.titleFrame.setVisibility(View.VISIBLE);
  175. }
  176. }
  177. // Setup content
  178. if (dialog.content != null) {
  179. dialog.content.setMovementMethod(new LinkMovementMethod());
  180. dialog.setTypeface(dialog.content, builder.regularFont);
  181. dialog.content.setLineSpacing(0f, builder.contentLineSpacingMultiplier);
  182. if (builder.positiveColor == null)
  183. dialog.content.setLinkTextColor(DialogUtils.resolveColor(dialog.getContext(), android.R.attr.textColorPrimary));
  184. else
  185. dialog.content.setLinkTextColor(builder.positiveColor);
  186. dialog.content.setTextColor(builder.contentColor);
  187. dialog.content.setGravity(builder.contentGravity.getGravityInt());
  188. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
  189. //noinspection ResourceType
  190. dialog.content.setTextAlignment(builder.contentGravity.getTextAlignment());
  191. }
  192. if (builder.content != null) {
  193. dialog.content.setText(builder.content);
  194. dialog.content.setVisibility(View.VISIBLE);
  195. } else {
  196. dialog.content.setVisibility(View.GONE);
  197. }
  198. }
  199. // Setup action buttons
  200. dialog.view.setButtonGravity(builder.buttonsGravity);
  201. dialog.view.setButtonStackedGravity(builder.btnStackedGravity);
  202. dialog.view.setForceStack(builder.forceStacking);
  203. boolean textAllCaps;
  204. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
  205. textAllCaps = DialogUtils.resolveBoolean(builder.context, android.R.attr.textAllCaps, true);
  206. if (textAllCaps)
  207. textAllCaps = DialogUtils.resolveBoolean(builder.context, R.attr.textAllCaps, true);
  208. } else {
  209. textAllCaps = DialogUtils.resolveBoolean(builder.context, R.attr.textAllCaps, true);
  210. }
  211. MDButton positiveTextView = dialog.positiveButton;
  212. dialog.setTypeface(positiveTextView, builder.mediumFont);
  213. positiveTextView.setAllCapsCompat(textAllCaps);
  214. positiveTextView.setText(builder.positiveText);
  215. positiveTextView.setTextColor(builder.positiveColor);
  216. dialog.positiveButton.setStackedSelector(dialog.getButtonSelector(DialogAction.POSITIVE, true));
  217. dialog.positiveButton.setDefaultSelector(dialog.getButtonSelector(DialogAction.POSITIVE, false));
  218. dialog.positiveButton.setTag(DialogAction.POSITIVE);
  219. dialog.positiveButton.setOnClickListener(dialog);
  220. dialog.positiveButton.setVisibility(View.VISIBLE);
  221. MDButton negativeTextView = dialog.negativeButton;
  222. dialog.setTypeface(negativeTextView, builder.mediumFont);
  223. negativeTextView.setAllCapsCompat(textAllCaps);
  224. negativeTextView.setText(builder.negativeText);
  225. negativeTextView.setTextColor(builder.negativeColor);
  226. dialog.negativeButton.setStackedSelector(dialog.getButtonSelector(DialogAction.NEGATIVE, true));
  227. dialog.negativeButton.setDefaultSelector(dialog.getButtonSelector(DialogAction.NEGATIVE, false));
  228. dialog.negativeButton.setTag(DialogAction.NEGATIVE);
  229. dialog.negativeButton.setOnClickListener(dialog);
  230. dialog.negativeButton.setVisibility(View.VISIBLE);
  231. MDButton neutralTextView = dialog.neutralButton;
  232. dialog.setTypeface(neutralTextView, builder.mediumFont);
  233. neutralTextView.setAllCapsCompat(textAllCaps);
  234. neutralTextView.setText(builder.neutralText);
  235. neutralTextView.setTextColor(builder.neutralColor);
  236. dialog.neutralButton.setStackedSelector(dialog.getButtonSelector(DialogAction.NEUTRAL, true));
  237. dialog.neutralButton.setDefaultSelector(dialog.getButtonSelector(DialogAction.NEUTRAL, false));
  238. dialog.neutralButton.setTag(DialogAction.NEUTRAL);
  239. dialog.neutralButton.setOnClickListener(dialog);
  240. dialog.neutralButton.setVisibility(View.VISIBLE);
  241. // Setup list dialog stuff
  242. if (builder.listCallbackMultiChoice != null)
  243. dialog.selectedIndicesList = new ArrayList<>();
  244. if (dialog.listView != null && (builder.items != null && builder.items.length > 0 || builder.adapter != null)) {
  245. dialog.listView.setSelector(dialog.getListSelector());
  246. // No custom adapter specified, setup the list with a MaterialDialogAdapter.
  247. // Which supports regular lists and single/multi choice dialogs.
  248. if (builder.adapter == null) {
  249. // Determine list type
  250. if (builder.listCallbackSingleChoice != null) {
  251. dialog.listType = MaterialDialog.ListType.SINGLE;
  252. } else if (builder.listCallbackMultiChoice != null) {
  253. dialog.listType = MaterialDialog.ListType.MULTI;
  254. if (builder.selectedIndices != null)
  255. dialog.selectedIndicesList = new ArrayList<>(Arrays.asList(builder.selectedIndices));
  256. } else {
  257. dialog.listType = MaterialDialog.ListType.REGULAR;
  258. }
  259. builder.adapter = new MaterialDialogAdapter(dialog,
  260. MaterialDialog.ListType.getLayoutForType(dialog.listType));
  261. } else if (builder.adapter instanceof MaterialSimpleListAdapter) {
  262. // Notify simple list adapter of the dialog it belongs to
  263. ((MaterialSimpleListAdapter) builder.adapter).setDialog(dialog, false);
  264. }
  265. }
  266. // Setup progress dialog stuff if needed
  267. setupProgressDialog(dialog);
  268. // Setup input dialog stuff if needed
  269. setupInputDialog(dialog);
  270. // Setup custom views
  271. if (builder.customView != null) {
  272. ((MDRootLayout) dialog.view.findViewById(R.id.root)).noTitleNoPadding();
  273. FrameLayout frame = (FrameLayout) dialog.view.findViewById(R.id.customViewFrame);
  274. dialog.customViewFrame = frame;
  275. View innerView = builder.customView;
  276. if (builder.wrapCustomViewInScroll) {
  277. /* Apply the frame padding to the content, this allows the ScrollView to draw it's
  278. over scroll glow without clipping */
  279. final Resources r = dialog.getContext().getResources();
  280. final int framePadding = r.getDimensionPixelSize(R.dimen.md_dialog_frame_margin);
  281. final ScrollView sv = new ScrollView(dialog.getContext());
  282. int paddingTop = r.getDimensionPixelSize(R.dimen.md_content_padding_top);
  283. int paddingBottom = r.getDimensionPixelSize(R.dimen.md_content_padding_bottom);
  284. sv.setClipToPadding(false);
  285. if (innerView instanceof EditText) {
  286. // Setting padding to an EditText causes visual errors, set it to the parent instead
  287. sv.setPadding(framePadding, paddingTop, framePadding, paddingBottom);
  288. } else {
  289. // Setting padding to scroll view pushes the scroll bars out, don't do it if not necessary (like above)
  290. sv.setPadding(0, paddingTop, 0, paddingBottom);
  291. innerView.setPadding(framePadding, 0, framePadding, 0);
  292. }
  293. sv.addView(innerView, new ScrollView.LayoutParams(
  294. ViewGroup.LayoutParams.MATCH_PARENT,
  295. ViewGroup.LayoutParams.WRAP_CONTENT));
  296. innerView = sv;
  297. }
  298. frame.addView(innerView, new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
  299. ViewGroup.LayoutParams.WRAP_CONTENT));
  300. }
  301. // Setup user listeners
  302. if (builder.showListener != null)
  303. dialog.setOnShowListener(builder.showListener);
  304. if (builder.cancelListener != null)
  305. dialog.setOnCancelListener(builder.cancelListener);
  306. if (builder.dismissListener != null)
  307. dialog.setOnDismissListener(builder.dismissListener);
  308. if (builder.keyListener != null)
  309. dialog.setOnKeyListener(builder.keyListener);
  310. // Setup internal show listener
  311. dialog.setOnShowListenerInternal();
  312. // Other internal initialization
  313. dialog.invalidateList();
  314. dialog.setViewInternal(dialog.view);
  315. dialog.checkIfListInitScroll();
  316. }
  317. private static void setupProgressDialog(final MaterialDialog dialog) {
  318. final MaterialDialog.Builder builder = dialog.mBuilder;
  319. if (builder.indeterminateProgress || builder.progress > -2) {
  320. dialog.mProgress = (ProgressBar) dialog.view.findViewById(android.R.id.progress);
  321. if (dialog.mProgress == null) return;
  322. if (builder.indeterminateProgress && !builder.indeterminateIsHorizontalProgress &&
  323. Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH &&
  324. Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
  325. dialog.mProgress.setIndeterminateDrawable(new CircularProgressDrawable(
  326. builder.widgetColor, builder.context.getResources().getDimension(R.dimen.circular_progress_border)));
  327. MDTintHelper.setTint(dialog.mProgress, builder.widgetColor, true);
  328. } else {
  329. MDTintHelper.setTint(dialog.mProgress, builder.widgetColor);
  330. }
  331. if (!builder.indeterminateProgress || builder.indeterminateIsHorizontalProgress) {
  332. dialog.mProgress.setIndeterminate(builder.indeterminateIsHorizontalProgress);
  333. dialog.mProgress.setProgress(0);
  334. dialog.mProgress.setMax(builder.progressMax);
  335. dialog.mProgressLabel = (TextView) dialog.view.findViewById(R.id.label);
  336. if (dialog.mProgressLabel != null) {
  337. dialog.mProgressLabel.setTextColor(builder.contentColor);
  338. dialog.setTypeface(dialog.mProgressLabel, builder.mediumFont);
  339. dialog.mProgressLabel.setText(builder.progressPercentFormat.format(0));
  340. }
  341. dialog.mProgressMinMax = (TextView) dialog.view.findViewById(R.id.minMax);
  342. if (dialog.mProgressMinMax != null) {
  343. dialog.mProgressMinMax.setTextColor(builder.contentColor);
  344. dialog.setTypeface(dialog.mProgressMinMax, builder.regularFont);
  345. if (builder.showMinMax) {
  346. dialog.mProgressMinMax.setVisibility(View.VISIBLE);
  347. dialog.mProgressMinMax.setText(String.format(builder.progressNumberFormat,
  348. 0, builder.progressMax));
  349. ViewGroup.MarginLayoutParams lp = (ViewGroup.MarginLayoutParams) dialog.mProgress.getLayoutParams();
  350. lp.leftMargin = 0;
  351. lp.rightMargin = 0;
  352. } else {
  353. dialog.mProgressMinMax.setVisibility(View.GONE);
  354. }
  355. } else {
  356. builder.showMinMax = false;
  357. }
  358. }
  359. }
  360. }
  361. private static void setupInputDialog(final MaterialDialog dialog) {
  362. final MaterialDialog.Builder builder = dialog.mBuilder;
  363. dialog.input = (EditText) dialog.view.findViewById(android.R.id.input);
  364. if (dialog.input == null) return;
  365. dialog.setTypeface(dialog.input, builder.regularFont);
  366. if (builder.inputPrefill != null)
  367. dialog.input.setText(builder.inputPrefill);
  368. dialog.setInternalInputCallback();
  369. dialog.input.setHint(builder.inputHint);
  370. dialog.input.setSingleLine();
  371. dialog.input.setTextColor(builder.contentColor);
  372. dialog.input.setHintTextColor(DialogUtils.adjustAlpha(builder.contentColor, 0.3f));
  373. MDTintHelper.setTint(dialog.input, dialog.mBuilder.widgetColor);
  374. if (builder.inputType != -1) {
  375. dialog.input.setInputType(builder.inputType);
  376. if ((builder.inputType & InputType.TYPE_TEXT_VARIATION_PASSWORD) == InputType.TYPE_TEXT_VARIATION_PASSWORD) {
  377. // If the flags contain TYPE_TEXT_VARIATION_PASSWORD, apply the password transformation method automatically
  378. dialog.input.setTransformationMethod(PasswordTransformationMethod.getInstance());
  379. }
  380. }
  381. dialog.inputMinMax = (TextView) dialog.view.findViewById(R.id.minMax);
  382. if (builder.inputMaxLength > -1) {
  383. dialog.invalidateInputMinMaxIndicator(dialog.input.getText().toString().length(),
  384. !builder.inputAllowEmpty);
  385. } else {
  386. dialog.inputMinMax.setVisibility(View.GONE);
  387. dialog.inputMinMax = null;
  388. }
  389. }
  390. }