DialogInit.java 22 KB

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