/* * Copyright (C) 2007 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.app; import com.android.internal.R; import android.annotation.UnsupportedAppUsage; import android.content.Context; import android.content.res.TypedArray; import android.graphics.drawable.Drawable; import android.os.Build; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.text.Spannable; import android.text.SpannableString; import android.text.style.StyleSpan; import android.view.LayoutInflater; import android.view.View; import android.widget.ProgressBar; import android.widget.TextView; import java.text.NumberFormat; /** * A dialog showing a progress indicator and an optional text message or view. * Only a text message or a view can be used at the same time. * *

The dialog can be made cancelable on back key press.

* *

The progress range is 0 to {@link #getMax() max}.

* * @deprecated ProgressDialog is a modal dialog, which prevents the * user from interacting with the app. Instead of using this class, you should * use a progress indicator like {@link android.widget.ProgressBar}, which can * be embedded in your app's UI. Alternatively, you can use a * notification * to inform the user of the task's progress. */ @Deprecated public class ProgressDialog extends AlertDialog { /** * Creates a ProgressDialog with a circular, spinning progress * bar. This is the default. */ public static final int STYLE_SPINNER = 0; /** * Creates a ProgressDialog with a horizontal progress bar. */ public static final int STYLE_HORIZONTAL = 1; @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023) private ProgressBar mProgress; @UnsupportedAppUsage private TextView mMessageView; private int mProgressStyle = STYLE_SPINNER; @UnsupportedAppUsage private TextView mProgressNumber; private String mProgressNumberFormat; private TextView mProgressPercent; private NumberFormat mProgressPercentFormat; private int mMax; private int mProgressVal; private int mSecondaryProgressVal; private int mIncrementBy; private int mIncrementSecondaryBy; private Drawable mProgressDrawable; private Drawable mIndeterminateDrawable; private CharSequence mMessage; private boolean mIndeterminate; private boolean mHasStarted; private Handler mViewUpdateHandler; /** * Creates a Progress dialog. * * @param context the parent context */ public ProgressDialog(Context context) { super(context); initFormats(); } /** * Creates a Progress dialog. * * @param context the parent context * @param theme the resource ID of the theme against which to inflate * this dialog, or {@code 0} to use the parent * {@code context}'s default alert dialog theme */ public ProgressDialog(Context context, int theme) { super(context, theme); initFormats(); } private void initFormats() { mProgressNumberFormat = "%1d/%2d"; mProgressPercentFormat = NumberFormat.getPercentInstance(); mProgressPercentFormat.setMaximumFractionDigits(0); } /** * Creates and shows a ProgressDialog. * * @param context the parent context * @param title the title text for the dialog's window * @param message the text to be displayed in the dialog * @return the ProgressDialog */ public static ProgressDialog show(Context context, CharSequence title, CharSequence message) { return show(context, title, message, false); } /** * Creates and shows a ProgressDialog. * * @param context the parent context * @param title the title text for the dialog's window * @param message the text to be displayed in the dialog * @param indeterminate true if the dialog should be {@link #setIndeterminate(boolean) * indeterminate}, false otherwise * @return the ProgressDialog */ public static ProgressDialog show(Context context, CharSequence title, CharSequence message, boolean indeterminate) { return show(context, title, message, indeterminate, false, null); } /** * Creates and shows a ProgressDialog. * * @param context the parent context * @param title the title text for the dialog's window * @param message the text to be displayed in the dialog * @param indeterminate true if the dialog should be {@link #setIndeterminate(boolean) * indeterminate}, false otherwise * @param cancelable true if the dialog is {@link #setCancelable(boolean) cancelable}, * false otherwise * @return the ProgressDialog */ public static ProgressDialog show(Context context, CharSequence title, CharSequence message, boolean indeterminate, boolean cancelable) { return show(context, title, message, indeterminate, cancelable, null); } /** * Creates and shows a ProgressDialog. * * @param context the parent context * @param title the title text for the dialog's window * @param message the text to be displayed in the dialog * @param indeterminate true if the dialog should be {@link #setIndeterminate(boolean) * indeterminate}, false otherwise * @param cancelable true if the dialog is {@link #setCancelable(boolean) cancelable}, * false otherwise * @param cancelListener the {@link #setOnCancelListener(OnCancelListener) listener} * to be invoked when the dialog is canceled * @return the ProgressDialog */ public static ProgressDialog show(Context context, CharSequence title, CharSequence message, boolean indeterminate, boolean cancelable, OnCancelListener cancelListener) { ProgressDialog dialog = new ProgressDialog(context); dialog.setTitle(title); dialog.setMessage(message); dialog.setIndeterminate(indeterminate); dialog.setCancelable(cancelable); dialog.setOnCancelListener(cancelListener); dialog.show(); return dialog; } @Override protected void onCreate(Bundle savedInstanceState) { LayoutInflater inflater = LayoutInflater.from(mContext); TypedArray a = mContext.obtainStyledAttributes(null, com.android.internal.R.styleable.AlertDialog, com.android.internal.R.attr.alertDialogStyle, 0); if (mProgressStyle == STYLE_HORIZONTAL) { /* Use a separate handler to update the text views as they * must be updated on the same thread that created them. */ mViewUpdateHandler = new Handler() { @Override public void handleMessage(Message msg) { super.handleMessage(msg); /* Update the number and percent */ int progress = mProgress.getProgress(); int max = mProgress.getMax(); if (mProgressNumberFormat != null) { String format = mProgressNumberFormat; mProgressNumber.setText(String.format(format, progress, max)); } else { mProgressNumber.setText(""); } if (mProgressPercentFormat != null) { double percent = (double) progress / (double) max; SpannableString tmp = new SpannableString(mProgressPercentFormat.format(percent)); tmp.setSpan(new StyleSpan(android.graphics.Typeface.BOLD), 0, tmp.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE); mProgressPercent.setText(tmp); } else { mProgressPercent.setText(""); } } }; View view = inflater.inflate(a.getResourceId( com.android.internal.R.styleable.AlertDialog_horizontalProgressLayout, R.layout.alert_dialog_progress), null); mProgress = (ProgressBar) view.findViewById(R.id.progress); mProgressNumber = (TextView) view.findViewById(R.id.progress_number); mProgressPercent = (TextView) view.findViewById(R.id.progress_percent); setView(view); } else { View view = inflater.inflate(a.getResourceId( com.android.internal.R.styleable.AlertDialog_progressLayout, R.layout.progress_dialog), null); mProgress = (ProgressBar) view.findViewById(R.id.progress); mMessageView = (TextView) view.findViewById(R.id.message); setView(view); } a.recycle(); if (mMax > 0) { setMax(mMax); } if (mProgressVal > 0) { setProgress(mProgressVal); } if (mSecondaryProgressVal > 0) { setSecondaryProgress(mSecondaryProgressVal); } if (mIncrementBy > 0) { incrementProgressBy(mIncrementBy); } if (mIncrementSecondaryBy > 0) { incrementSecondaryProgressBy(mIncrementSecondaryBy); } if (mProgressDrawable != null) { setProgressDrawable(mProgressDrawable); } if (mIndeterminateDrawable != null) { setIndeterminateDrawable(mIndeterminateDrawable); } if (mMessage != null) { setMessage(mMessage); } setIndeterminate(mIndeterminate); onProgressChanged(); super.onCreate(savedInstanceState); } @Override public void onStart() { super.onStart(); mHasStarted = true; } @Override protected void onStop() { super.onStop(); mHasStarted = false; } /** * Sets the current progress. * * @param value the current progress, a value between 0 and {@link #getMax()} * * @see ProgressBar#setProgress(int) */ public void setProgress(int value) { if (mHasStarted) { mProgress.setProgress(value); onProgressChanged(); } else { mProgressVal = value; } } /** * Sets the secondary progress. * * @param secondaryProgress the current secondary progress, a value between 0 and * {@link #getMax()} * * @see ProgressBar#setSecondaryProgress(int) */ public void setSecondaryProgress(int secondaryProgress) { if (mProgress != null) { mProgress.setSecondaryProgress(secondaryProgress); onProgressChanged(); } else { mSecondaryProgressVal = secondaryProgress; } } /** * Gets the current progress. * * @return the current progress, a value between 0 and {@link #getMax()} */ public int getProgress() { if (mProgress != null) { return mProgress.getProgress(); } return mProgressVal; } /** * Gets the current secondary progress. * * @return the current secondary progress, a value between 0 and {@link #getMax()} */ public int getSecondaryProgress() { if (mProgress != null) { return mProgress.getSecondaryProgress(); } return mSecondaryProgressVal; } /** * Gets the maximum allowed progress value. The default value is 100. * * @return the maximum value */ public int getMax() { if (mProgress != null) { return mProgress.getMax(); } return mMax; } /** * Sets the maximum allowed progress value. */ public void setMax(int max) { if (mProgress != null) { mProgress.setMax(max); onProgressChanged(); } else { mMax = max; } } /** * Increments the current progress value. * * @param diff the amount by which the current progress will be incremented, * up to {@link #getMax()} */ public void incrementProgressBy(int diff) { if (mProgress != null) { mProgress.incrementProgressBy(diff); onProgressChanged(); } else { mIncrementBy += diff; } } /** * Increments the current secondary progress value. * * @param diff the amount by which the current secondary progress will be incremented, * up to {@link #getMax()} */ public void incrementSecondaryProgressBy(int diff) { if (mProgress != null) { mProgress.incrementSecondaryProgressBy(diff); onProgressChanged(); } else { mIncrementSecondaryBy += diff; } } /** * Sets the drawable to be used to display the progress value. * * @param d the drawable to be used * * @see ProgressBar#setProgressDrawable(Drawable) */ public void setProgressDrawable(Drawable d) { if (mProgress != null) { mProgress.setProgressDrawable(d); } else { mProgressDrawable = d; } } /** * Sets the drawable to be used to display the indeterminate progress value. * * @param d the drawable to be used * * @see ProgressBar#setProgressDrawable(Drawable) * @see #setIndeterminate(boolean) */ public void setIndeterminateDrawable(Drawable d) { if (mProgress != null) { mProgress.setIndeterminateDrawable(d); } else { mIndeterminateDrawable = d; } } /** * Change the indeterminate mode for this ProgressDialog. In indeterminate * mode, the progress is ignored and the dialog shows an infinite * animation instead. * *

Note: A ProgressDialog with style {@link #STYLE_SPINNER} * is always indeterminate and will ignore this setting.

* * @param indeterminate true to enable indeterminate mode, false otherwise * * @see #setProgressStyle(int) */ public void setIndeterminate(boolean indeterminate) { if (mProgress != null) { mProgress.setIndeterminate(indeterminate); } else { mIndeterminate = indeterminate; } } /** * Whether this ProgressDialog is in indeterminate mode. * * @return true if the dialog is in indeterminate mode, false otherwise */ public boolean isIndeterminate() { if (mProgress != null) { return mProgress.isIndeterminate(); } return mIndeterminate; } @Override public void setMessage(CharSequence message) { if (mProgress != null) { if (mProgressStyle == STYLE_HORIZONTAL) { super.setMessage(message); } else { mMessageView.setText(message); } } else { mMessage = message; } } /** * Sets the style of this ProgressDialog, either {@link #STYLE_SPINNER} or * {@link #STYLE_HORIZONTAL}. The default is {@link #STYLE_SPINNER}. * *

Note: A ProgressDialog with style {@link #STYLE_SPINNER} * is always indeterminate and will ignore the {@link #setIndeterminate(boolean) * indeterminate} setting.

* * @param style the style of this ProgressDialog, either {@link #STYLE_SPINNER} or * {@link #STYLE_HORIZONTAL} */ public void setProgressStyle(int style) { mProgressStyle = style; } /** * Change the format of the small text showing current and maximum units * of progress. The default is "%1d/%2d". * Should not be called during the number is progressing. * @param format A string passed to {@link String#format String.format()}; * use "%1d" for the current number and "%2d" for the maximum. If null, * nothing will be shown. */ public void setProgressNumberFormat(String format) { mProgressNumberFormat = format; onProgressChanged(); } /** * Change the format of the small text showing the percentage of progress. * The default is * {@link NumberFormat#getPercentInstance() NumberFormat.getPercentageInstnace().} * Should not be called during the number is progressing. * @param format An instance of a {@link NumberFormat} to generate the * percentage text. If null, nothing will be shown. */ public void setProgressPercentFormat(NumberFormat format) { mProgressPercentFormat = format; onProgressChanged(); } private void onProgressChanged() { if (mProgressStyle == STYLE_HORIZONTAL) { if (mViewUpdateHandler != null && !mViewUpdateHandler.hasMessages(0)) { mViewUpdateHandler.sendEmptyMessage(0); } } } }