DialogInit.java 22 KB

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