1/*
2 * Copyright (C) 2007 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package android.app;
18
19import com.android.internal.R;
20
21import android.content.Context;
22import android.content.res.TypedArray;
23import android.graphics.drawable.Drawable;
24import android.os.Bundle;
25import android.os.Handler;
26import android.os.Message;
27import android.text.Spannable;
28import android.text.SpannableString;
29import android.text.style.StyleSpan;
30import android.view.LayoutInflater;
31import android.view.View;
32import android.widget.ProgressBar;
33import android.widget.TextView;
34
35import java.text.NumberFormat;
36
37/**
38 * A dialog showing a progress indicator and an optional text message or view.
39 * Only a text message or a view can be used at the same time.
40 *
41 * <p>The dialog can be made cancelable on back key press.</p>
42 *
43 * <p>The progress range is 0 to {@link #getMax() max}.</p>
44 *
45 * @deprecated <code>ProgressDialog</code> is a modal dialog, which prevents the
46 * user from interacting with the app. Instead of using this class, you should
47 * use a progress indicator like {@link android.widget.ProgressBar}, which can
48 * be embedded in your app's UI. Alternatively, you can use a
49 * <a href="/guide/topics/ui/notifiers/notifications.html">notification</a>
50 * to inform the user of the task's progress.
51 */
52@Deprecated
53public class ProgressDialog extends AlertDialog {
54
55    /**
56     * Creates a ProgressDialog with a circular, spinning progress
57     * bar. This is the default.
58     */
59    public static final int STYLE_SPINNER = 0;
60
61    /**
62     * Creates a ProgressDialog with a horizontal progress bar.
63     */
64    public static final int STYLE_HORIZONTAL = 1;
65
66    private ProgressBar mProgress;
67    private TextView mMessageView;
68
69    private int mProgressStyle = STYLE_SPINNER;
70    private TextView mProgressNumber;
71    private String mProgressNumberFormat;
72    private TextView mProgressPercent;
73    private NumberFormat mProgressPercentFormat;
74
75    private int mMax;
76    private int mProgressVal;
77    private int mSecondaryProgressVal;
78    private int mIncrementBy;
79    private int mIncrementSecondaryBy;
80    private Drawable mProgressDrawable;
81    private Drawable mIndeterminateDrawable;
82    private CharSequence mMessage;
83    private boolean mIndeterminate;
84
85    private boolean mHasStarted;
86    private Handler mViewUpdateHandler;
87
88    /**
89     * Creates a Progress dialog.
90     *
91     * @param context the parent context
92     */
93    public ProgressDialog(Context context) {
94        super(context);
95        initFormats();
96    }
97
98    /**
99     * Creates a Progress dialog.
100     *
101     * @param context the parent context
102     * @param theme the resource ID of the theme against which to inflate
103     *              this dialog, or {@code 0} to use the parent
104     *              {@code context}'s default alert dialog theme
105     */
106    public ProgressDialog(Context context, int theme) {
107        super(context, theme);
108        initFormats();
109    }
110
111    private void initFormats() {
112        mProgressNumberFormat = "%1d/%2d";
113        mProgressPercentFormat = NumberFormat.getPercentInstance();
114        mProgressPercentFormat.setMaximumFractionDigits(0);
115    }
116
117    /**
118     * Creates and shows a ProgressDialog.
119     *
120     * @param context the parent context
121     * @param title the title text for the dialog's window
122     * @param message the text to be displayed in the dialog
123     * @return the ProgressDialog
124     */
125    public static ProgressDialog show(Context context, CharSequence title,
126            CharSequence message) {
127        return show(context, title, message, false);
128    }
129
130    /**
131     * Creates and shows a ProgressDialog.
132     *
133     * @param context the parent context
134     * @param title the title text for the dialog's window
135     * @param message the text to be displayed in the dialog
136     * @param indeterminate true if the dialog should be {@link #setIndeterminate(boolean)
137     *        indeterminate}, false otherwise
138     * @return the ProgressDialog
139     */
140    public static ProgressDialog show(Context context, CharSequence title,
141            CharSequence message, boolean indeterminate) {
142        return show(context, title, message, indeterminate, false, null);
143    }
144
145    /**
146     * Creates and shows a ProgressDialog.
147     *
148     * @param context the parent context
149     * @param title the title text for the dialog's window
150     * @param message the text to be displayed in the dialog
151     * @param indeterminate true if the dialog should be {@link #setIndeterminate(boolean)
152     *        indeterminate}, false otherwise
153     * @param cancelable true if the dialog is {@link #setCancelable(boolean) cancelable},
154     *        false otherwise
155     * @return the ProgressDialog
156     */
157    public static ProgressDialog show(Context context, CharSequence title,
158            CharSequence message, boolean indeterminate, boolean cancelable) {
159        return show(context, title, message, indeterminate, cancelable, null);
160    }
161
162    /**
163     * Creates and shows a ProgressDialog.
164     *
165     * @param context the parent context
166     * @param title the title text for the dialog's window
167     * @param message the text to be displayed in the dialog
168     * @param indeterminate true if the dialog should be {@link #setIndeterminate(boolean)
169     *        indeterminate}, false otherwise
170     * @param cancelable true if the dialog is {@link #setCancelable(boolean) cancelable},
171     *        false otherwise
172     * @param cancelListener the {@link #setOnCancelListener(OnCancelListener) listener}
173     *        to be invoked when the dialog is canceled
174     * @return the ProgressDialog
175     */
176    public static ProgressDialog show(Context context, CharSequence title,
177            CharSequence message, boolean indeterminate,
178            boolean cancelable, OnCancelListener cancelListener) {
179        ProgressDialog dialog = new ProgressDialog(context);
180        dialog.setTitle(title);
181        dialog.setMessage(message);
182        dialog.setIndeterminate(indeterminate);
183        dialog.setCancelable(cancelable);
184        dialog.setOnCancelListener(cancelListener);
185        dialog.show();
186        return dialog;
187    }
188
189    @Override
190    protected void onCreate(Bundle savedInstanceState) {
191        LayoutInflater inflater = LayoutInflater.from(mContext);
192        TypedArray a = mContext.obtainStyledAttributes(null,
193                com.android.internal.R.styleable.AlertDialog,
194                com.android.internal.R.attr.alertDialogStyle, 0);
195        if (mProgressStyle == STYLE_HORIZONTAL) {
196
197            /* Use a separate handler to update the text views as they
198             * must be updated on the same thread that created them.
199             */
200            mViewUpdateHandler = new Handler() {
201                @Override
202                public void handleMessage(Message msg) {
203                    super.handleMessage(msg);
204
205                    /* Update the number and percent */
206                    int progress = mProgress.getProgress();
207                    int max = mProgress.getMax();
208                    if (mProgressNumberFormat != null) {
209                        String format = mProgressNumberFormat;
210                        mProgressNumber.setText(String.format(format, progress, max));
211                    } else {
212                        mProgressNumber.setText("");
213                    }
214                    if (mProgressPercentFormat != null) {
215                        double percent = (double) progress / (double) max;
216                        SpannableString tmp = new SpannableString(mProgressPercentFormat.format(percent));
217                        tmp.setSpan(new StyleSpan(android.graphics.Typeface.BOLD),
218                                0, tmp.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
219                        mProgressPercent.setText(tmp);
220                    } else {
221                        mProgressPercent.setText("");
222                    }
223                }
224            };
225            View view = inflater.inflate(a.getResourceId(
226                    com.android.internal.R.styleable.AlertDialog_horizontalProgressLayout,
227                    R.layout.alert_dialog_progress), null);
228            mProgress = (ProgressBar) view.findViewById(R.id.progress);
229            mProgressNumber = (TextView) view.findViewById(R.id.progress_number);
230            mProgressPercent = (TextView) view.findViewById(R.id.progress_percent);
231            setView(view);
232        } else {
233            View view = inflater.inflate(a.getResourceId(
234                    com.android.internal.R.styleable.AlertDialog_progressLayout,
235                    R.layout.progress_dialog), null);
236            mProgress = (ProgressBar) view.findViewById(R.id.progress);
237            mMessageView = (TextView) view.findViewById(R.id.message);
238            setView(view);
239        }
240        a.recycle();
241        if (mMax > 0) {
242            setMax(mMax);
243        }
244        if (mProgressVal > 0) {
245            setProgress(mProgressVal);
246        }
247        if (mSecondaryProgressVal > 0) {
248            setSecondaryProgress(mSecondaryProgressVal);
249        }
250        if (mIncrementBy > 0) {
251            incrementProgressBy(mIncrementBy);
252        }
253        if (mIncrementSecondaryBy > 0) {
254            incrementSecondaryProgressBy(mIncrementSecondaryBy);
255        }
256        if (mProgressDrawable != null) {
257            setProgressDrawable(mProgressDrawable);
258        }
259        if (mIndeterminateDrawable != null) {
260            setIndeterminateDrawable(mIndeterminateDrawable);
261        }
262        if (mMessage != null) {
263            setMessage(mMessage);
264        }
265        setIndeterminate(mIndeterminate);
266        onProgressChanged();
267        super.onCreate(savedInstanceState);
268    }
269
270    @Override
271    public void onStart() {
272        super.onStart();
273        mHasStarted = true;
274    }
275
276    @Override
277    protected void onStop() {
278        super.onStop();
279        mHasStarted = false;
280    }
281
282    /**
283     * Sets the current progress.
284     *
285     * @param value the current progress, a value between 0 and {@link #getMax()}
286     *
287     * @see ProgressBar#setProgress(int)
288     */
289    public void setProgress(int value) {
290        if (mHasStarted) {
291            mProgress.setProgress(value);
292            onProgressChanged();
293        } else {
294            mProgressVal = value;
295        }
296    }
297
298    /**
299     * Sets the secondary progress.
300     *
301     * @param secondaryProgress the current secondary progress, a value between 0 and
302     * {@link #getMax()}
303     *
304     * @see ProgressBar#setSecondaryProgress(int)
305     */
306    public void setSecondaryProgress(int secondaryProgress) {
307        if (mProgress != null) {
308            mProgress.setSecondaryProgress(secondaryProgress);
309            onProgressChanged();
310        } else {
311            mSecondaryProgressVal = secondaryProgress;
312        }
313    }
314
315    /**
316     * Gets the current progress.
317     *
318     * @return the current progress, a value between 0 and {@link #getMax()}
319     */
320    public int getProgress() {
321        if (mProgress != null) {
322            return mProgress.getProgress();
323        }
324        return mProgressVal;
325    }
326
327    /**
328     * Gets the current secondary progress.
329     *
330     * @return the current secondary progress, a value between 0 and {@link #getMax()}
331     */
332    public int getSecondaryProgress() {
333        if (mProgress != null) {
334            return mProgress.getSecondaryProgress();
335        }
336        return mSecondaryProgressVal;
337    }
338
339    /**
340     * Gets the maximum allowed progress value. The default value is 100.
341     *
342     * @return the maximum value
343     */
344    public int getMax() {
345        if (mProgress != null) {
346            return mProgress.getMax();
347        }
348        return mMax;
349    }
350
351    /**
352     * Sets the maximum allowed progress value.
353     */
354    public void setMax(int max) {
355        if (mProgress != null) {
356            mProgress.setMax(max);
357            onProgressChanged();
358        } else {
359            mMax = max;
360        }
361    }
362
363    /**
364     * Increments the current progress value.
365     *
366     * @param diff the amount by which the current progress will be incremented,
367     * up to {@link #getMax()}
368     */
369    public void incrementProgressBy(int diff) {
370        if (mProgress != null) {
371            mProgress.incrementProgressBy(diff);
372            onProgressChanged();
373        } else {
374            mIncrementBy += diff;
375        }
376    }
377
378    /**
379     * Increments the current secondary progress value.
380     *
381     * @param diff the amount by which the current secondary progress will be incremented,
382     * up to {@link #getMax()}
383     */
384    public void incrementSecondaryProgressBy(int diff) {
385        if (mProgress != null) {
386            mProgress.incrementSecondaryProgressBy(diff);
387            onProgressChanged();
388        } else {
389            mIncrementSecondaryBy += diff;
390        }
391    }
392
393    /**
394     * Sets the drawable to be used to display the progress value.
395     *
396     * @param d the drawable to be used
397     *
398     * @see ProgressBar#setProgressDrawable(Drawable)
399     */
400    public void setProgressDrawable(Drawable d) {
401        if (mProgress != null) {
402            mProgress.setProgressDrawable(d);
403        } else {
404            mProgressDrawable = d;
405        }
406    }
407
408    /**
409     * Sets the drawable to be used to display the indeterminate progress value.
410     *
411     * @param d the drawable to be used
412     *
413     * @see ProgressBar#setProgressDrawable(Drawable)
414     * @see #setIndeterminate(boolean)
415     */
416    public void setIndeterminateDrawable(Drawable d) {
417        if (mProgress != null) {
418            mProgress.setIndeterminateDrawable(d);
419        } else {
420            mIndeterminateDrawable = d;
421        }
422    }
423
424    /**
425     * Change the indeterminate mode for this ProgressDialog. In indeterminate
426     * mode, the progress is ignored and the dialog shows an infinite
427     * animation instead.
428     *
429     * <p><strong>Note:</strong> A ProgressDialog with style {@link #STYLE_SPINNER}
430     * is always indeterminate and will ignore this setting.</p>
431     *
432     * @param indeterminate true to enable indeterminate mode, false otherwise
433     *
434     * @see #setProgressStyle(int)
435     */
436    public void setIndeterminate(boolean indeterminate) {
437        if (mProgress != null) {
438            mProgress.setIndeterminate(indeterminate);
439        } else {
440            mIndeterminate = indeterminate;
441        }
442    }
443
444    /**
445     * Whether this ProgressDialog is in indeterminate mode.
446     *
447     * @return true if the dialog is in indeterminate mode, false otherwise
448     */
449    public boolean isIndeterminate() {
450        if (mProgress != null) {
451            return mProgress.isIndeterminate();
452        }
453        return mIndeterminate;
454    }
455
456    @Override
457    public void setMessage(CharSequence message) {
458        if (mProgress != null) {
459            if (mProgressStyle == STYLE_HORIZONTAL) {
460                super.setMessage(message);
461            } else {
462                mMessageView.setText(message);
463            }
464        } else {
465            mMessage = message;
466        }
467    }
468
469    /**
470     * Sets the style of this ProgressDialog, either {@link #STYLE_SPINNER} or
471     * {@link #STYLE_HORIZONTAL}. The default is {@link #STYLE_SPINNER}.
472     *
473     * <p><strong>Note:</strong> A ProgressDialog with style {@link #STYLE_SPINNER}
474     * is always indeterminate and will ignore the {@link #setIndeterminate(boolean)
475     * indeterminate} setting.</p>
476     *
477     * @param style the style of this ProgressDialog, either {@link #STYLE_SPINNER} or
478     * {@link #STYLE_HORIZONTAL}
479     */
480    public void setProgressStyle(int style) {
481        mProgressStyle = style;
482    }
483
484    /**
485     * Change the format of the small text showing current and maximum units
486     * of progress.  The default is "%1d/%2d".
487     * Should not be called during the number is progressing.
488     * @param format A string passed to {@link String#format String.format()};
489     * use "%1d" for the current number and "%2d" for the maximum.  If null,
490     * nothing will be shown.
491     */
492    public void setProgressNumberFormat(String format) {
493        mProgressNumberFormat = format;
494        onProgressChanged();
495    }
496
497    /**
498     * Change the format of the small text showing the percentage of progress.
499     * The default is
500     * {@link NumberFormat#getPercentInstance() NumberFormat.getPercentageInstnace().}
501     * Should not be called during the number is progressing.
502     * @param format An instance of a {@link NumberFormat} to generate the
503     * percentage text.  If null, nothing will be shown.
504     */
505    public void setProgressPercentFormat(NumberFormat format) {
506        mProgressPercentFormat = format;
507        onProgressChanged();
508    }
509
510    private void onProgressChanged() {
511        if (mProgressStyle == STYLE_HORIZONTAL) {
512            if (mViewUpdateHandler != null && !mViewUpdateHandler.hasMessages(0)) {
513                mViewUpdateHandler.sendEmptyMessage(0);
514            }
515        }
516    }
517}
518