Shared.java revision 5b0a2c187a9e446b683687817d22cbe443585223
1fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay/*
2fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay * Copyright (C) 2015 The Android Open Source Project
3fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay *
4fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay * Licensed under the Apache License, Version 2.0 (the "License");
5fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay * you may not use this file except in compliance with the License.
6fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay * You may obtain a copy of the License at
7fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay *
8fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay *      http://www.apache.org/licenses/LICENSE-2.0
9fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay *
10fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay * Unless required by applicable law or agreed to in writing, software
11fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay * distributed under the License is distributed on an "AS IS" BASIS,
12fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay * See the License for the specific language governing permissions and
14fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay * limitations under the License.
15fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay */
16fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay
17d9caa6ab53aa784acaf241c0ded3c4ae2d342bf8Steve McKaypackage com.android.documentsui.base;
18fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay
195b0a2c187a9e446b683687817d22cbe443585223Steve McKayimport android.annotation.PluralsRes;
2024917427f41f9e41925c4a56d0c71d85e30643dfSteve McKayimport android.app.Activity;
2117f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKayimport android.app.AlertDialog;
22919231857d2add3afe51f06aaf41663a252c3e0eBen Kwaimport android.content.Context;
2317f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKayimport android.content.Intent;
24988d8a354b00dce7e24deee187c08a4591956ac9Steve McKayimport android.content.pm.ApplicationInfo;
25741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronskaimport android.content.res.Configuration;
26988d8a354b00dce7e24deee187c08a4591956ac9Steve McKayimport android.net.Uri;
272056205cb85d529d732638c45fe459168168e1eeGarfield, Tanimport android.os.Looper;
2817f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKayimport android.provider.DocumentsContract;
2955c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKayimport android.text.TextUtils;
308e3fd7676023738c04d099f2940a635ff0699717Ben Kwaimport android.text.format.DateUtils;
318e3fd7676023738c04d099f2940a635ff0699717Ben Kwaimport android.text.format.Time;
322056205cb85d529d732638c45fe459168168e1eeGarfield, Tanimport android.util.Log;
33741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronskaimport android.view.WindowManager;
346d50bcc90e6f8b3c16e23b3fc2d63f57804dd805Aga Wronska
35d9caa6ab53aa784acaf241c0ded3c4ae2d342bf8Steve McKayimport com.android.documentsui.R;
36d9caa6ab53aa784acaf241c0ded3c4ae2d342bf8Steve McKay
3755c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKayimport java.text.Collator;
38c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKayimport java.util.ArrayList;
39c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKayimport java.util.List;
40c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKay
4185ec0d676297724f211213c7cb188839f1d3601bSteve McKayimport javax.annotation.Nullable;
4285ec0d676297724f211213c7cb188839f1d3601bSteve McKay
434d0255f79cc92a5675d14b20f9cdf06ecb8d7109Steve McKay/** @hide */
44fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKaypublic final class Shared {
4555c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay
460af8afd3309538dec784ed0c9c35b252a8213123Steve McKay    public static final String TAG = "Documents";
470af8afd3309538dec784ed0c9c35b252a8213123Steve McKay
482e809a14468df6437ccd1b147b421553e6a06a7eGarfield, Tan    public static final boolean DEBUG = true;
490af8afd3309538dec784ed0c9c35b252a8213123Steve McKay
50ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa    /** Intent action name to pick a copy destination. */
51ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa    public static final String ACTION_PICK_COPY_DESTINATION =
52ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa            "com.android.documentsui.PICK_COPY_DESTINATION";
53ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa
54ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa    /**
5517f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     * Extra flag allowing app to be opened in productivity mode (less downloadsy).
5617f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     * Useful developers and the likes. When set to true overrides the default
5717f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     * config value of productivity_device.
5817f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     */
5917f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    public static final String EXTRA_PRODUCTIVITY_MODE = "com.android.documentsui.PRODUCTIVITY";
6017f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay
6117f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    /**
62ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa     * Extra boolean flag for {@link ACTION_PICK_COPY_DESTINATION}, which
63ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa     * specifies if the destination directory needs to create new directory or not.
64ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa     */
65ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa    public static final String EXTRA_DIRECTORY_COPY = "com.android.documentsui.DIRECTORY_COPY";
6617f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay
6717f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    /**
6817f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     * Extra flag used to store the current stack so user opens in right spot.
6917f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     */
70af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    public static final String EXTRA_STACK = "com.android.documentsui.STACK";
71af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska
72af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    /**
73af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     * Extra flag used to store query of type String in the bundle.
74af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     */
75af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    public static final String EXTRA_QUERY = "query";
76af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska
77af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    /**
78af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     * Extra flag used to store state of type State in the bundle.
79af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     */
80af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    public static final String EXTRA_STATE = "state";
81af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska
82af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    /**
83af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     * Extra flag used to store type of DirectoryFragment's type ResultType type in the bundle.
84af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     */
85af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    public static final String EXTRA_TYPE = "type";
86af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska
87af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    /**
88af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     * Extra flag used to store root of type RootInfo in the bundle.
89af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     */
90af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    public static final String EXTRA_ROOT = "root";
91af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska
92af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    /**
93af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     * Extra flag used to store document of DocumentInfo type in the bundle.
94af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     */
95af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    public static final String EXTRA_DOC = "document";
96af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska
97af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    /**
98af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     * Extra flag used to store DirectoryFragment's selection of Selection type in the bundle.
99af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     */
100af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    public static final String EXTRA_SELECTION = "selection";
101af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska
102af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    /**
103af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     * Extra flag used to store DirectoryFragment's search mode of boolean type in the bundle.
104af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     */
105af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    public static final String EXTRA_SEARCH_MODE = "searchMode";
106af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska
107af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    /**
108af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     * Extra flag used to store DirectoryFragment's ignore state of boolean type in the bundle.
109af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska     */
110af5ace5d5e3a273b151e012206efd029c2872b59Aga Wronska    public static final String EXTRA_IGNORE_STATE = "ignoreState";
111ae96780a8909476dca709b88dbae5ec9a2f0632cBen Kwa
112b8373c22f1bacf194c35a45dde337cad2e904587Tomasz Mikolajewski    /**
113b8373c22f1bacf194c35a45dde337cad2e904587Tomasz Mikolajewski     * Extra for an Intent for enabling performance benchmark. Used only by tests.
114b8373c22f1bacf194c35a45dde337cad2e904587Tomasz Mikolajewski     */
115b8373c22f1bacf194c35a45dde337cad2e904587Tomasz Mikolajewski    public static final String EXTRA_BENCHMARK = "com.android.documentsui.benchmark";
116b8373c22f1bacf194c35a45dde337cad2e904587Tomasz Mikolajewski
1172ccad1e19bb5e9308ff06c90327de99cc509f613Tomasz Mikolajewski    /**
1182ccad1e19bb5e9308ff06c90327de99cc509f613Tomasz Mikolajewski     * Maximum number of items in a Binder transaction packet.
1192ccad1e19bb5e9308ff06c90327de99cc509f613Tomasz Mikolajewski     */
12084769b8205cf4a100c9d4f65c41cf72ba7a3b40fSteve McKay    public static final int MAX_DOCS_IN_INTENT = 500;
1212ccad1e19bb5e9308ff06c90327de99cc509f613Tomasz Mikolajewski
12283ac678e3ee261f3413486b4a2f9bd6337a820a8Steve McKay    /**
12383ac678e3ee261f3413486b4a2f9bd6337a820a8Steve McKay     * Animation duration of checkbox in directory list/grid in millis.
12483ac678e3ee261f3413486b4a2f9bd6337a820a8Steve McKay     */
12583ac678e3ee261f3413486b4a2f9bd6337a820a8Steve McKay    public static final int CHECK_ANIMATION_DURATION = 100;
12683ac678e3ee261f3413486b4a2f9bd6337a820a8Steve McKay
12755c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay    private static final Collator sCollator;
12855c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay
12955c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay    static {
13055c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay        sCollator = Collator.getInstance();
13155c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay        sCollator.setStrength(Collator.SECONDARY);
13255c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay    }
13355c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay
134919231857d2add3afe51f06aaf41663a252c3e0eBen Kwa    /**
1355b0a2c187a9e446b683687817d22cbe443585223Steve McKay     * @deprecated use {@ link MessageBuilder#getQuantityString}
136919231857d2add3afe51f06aaf41663a252c3e0eBen Kwa     */
1375b0a2c187a9e446b683687817d22cbe443585223Steve McKay    @Deprecated
1385b0a2c187a9e446b683687817d22cbe443585223Steve McKay    public static final String getQuantityString(Context context, @PluralsRes int resourceId, int quantity) {
139919231857d2add3afe51f06aaf41663a252c3e0eBen Kwa        return context.getResources().getQuantityString(resourceId, quantity, quantity);
140919231857d2add3afe51f06aaf41663a252c3e0eBen Kwa    }
1418e3fd7676023738c04d099f2940a635ff0699717Ben Kwa
1428e3fd7676023738c04d099f2940a635ff0699717Ben Kwa    public static String formatTime(Context context, long when) {
1438e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        // TODO: DateUtils should make this easier
1448e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        Time then = new Time();
1458e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        then.set(when);
1468e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        Time now = new Time();
1478e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        now.setToNow();
1488e3fd7676023738c04d099f2940a635ff0699717Ben Kwa
1498e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        int flags = DateUtils.FORMAT_NO_NOON | DateUtils.FORMAT_NO_MIDNIGHT
1508e3fd7676023738c04d099f2940a635ff0699717Ben Kwa                | DateUtils.FORMAT_ABBREV_ALL;
1518e3fd7676023738c04d099f2940a635ff0699717Ben Kwa
1528e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        if (then.year != now.year) {
1538e3fd7676023738c04d099f2940a635ff0699717Ben Kwa            flags |= DateUtils.FORMAT_SHOW_YEAR | DateUtils.FORMAT_SHOW_DATE;
1548e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        } else if (then.yearDay != now.yearDay) {
1558e3fd7676023738c04d099f2940a635ff0699717Ben Kwa            flags |= DateUtils.FORMAT_SHOW_DATE;
1568e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        } else {
1578e3fd7676023738c04d099f2940a635ff0699717Ben Kwa            flags |= DateUtils.FORMAT_SHOW_TIME;
1588e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        }
1598e3fd7676023738c04d099f2940a635ff0699717Ben Kwa
1608e3fd7676023738c04d099f2940a635ff0699717Ben Kwa        return DateUtils.formatDateTime(context, when, flags);
1618e3fd7676023738c04d099f2940a635ff0699717Ben Kwa    }
1628e3fd7676023738c04d099f2940a635ff0699717Ben Kwa
163c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKay    /**
164c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKay     * A convenient way to transform any list into a (parcelable) ArrayList.
165c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKay     * Uses cast if possible, else creates a new list with entries from {@code list}.
166c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKay     */
167c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKay    public static <T> ArrayList<T> asArrayList(List<T> list) {
168c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKay        return list instanceof ArrayList
169c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKay            ? (ArrayList<T>) list
170988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay            : new ArrayList<>(list);
171c83baa0574ee9e34c0e06bda1ff08928d880ee36Steve McKay    }
17255c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay
17355c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay    /**
17455c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay     * Compare two strings against each other using system default collator in a
17506b036f0236149aa8f19a3ededf3d66ba2121a8dTomasz Mikolajewski     * case-insensitive mode. Clusters strings prefixed with {@link DIR_PREFIX}
17606b036f0236149aa8f19a3ededf3d66ba2121a8dTomasz Mikolajewski     * before other items.
17755c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay     */
17855c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay    public static int compareToIgnoreCaseNullable(String lhs, String rhs) {
17955c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay        final boolean leftEmpty = TextUtils.isEmpty(lhs);
18055c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay        final boolean rightEmpty = TextUtils.isEmpty(rhs);
18155c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay
18255c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay        if (leftEmpty && rightEmpty) return 0;
18355c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay        if (leftEmpty) return -1;
18455c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay        if (rightEmpty) return 1;
18555c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay
18655c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay        return sCollator.compare(lhs, rhs);
18755c00e7356d9f76e7378cf7c701b9a41cb7be6daSteve McKay    }
188741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska
189988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay    /**
190988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay     * Returns the calling package, possibly overridden by EXTRA_PACKAGE_NAME.
191988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay     * @param activity
192988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay     * @return
193988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay     */
194988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay    public static String getCallingPackageName(Activity activity) {
195988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay        String callingPackage = activity.getCallingPackage();
196988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay        // System apps can set the calling package name using an extra.
197988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay        try {
198988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay            ApplicationInfo info =
199988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay                    activity.getPackageManager().getApplicationInfo(callingPackage, 0);
200988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay            if (info.isSystemApp() || info.isUpdatedSystemApp()) {
201988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay                final String extra = activity.getIntent().getStringExtra(
202988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay                        DocumentsContract.EXTRA_PACKAGE_NAME);
203988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay                if (extra != null) {
204988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay                    callingPackage = extra;
205988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay                }
206988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay            }
207988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay        } finally {
208988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay            return callingPackage;
209988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay        }
210988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay    }
211988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay
212988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay    /**
213988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay     * Returns the default directory to be presented after starting the activity.
214988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay     * Method can be overridden if the change of the behavior of the the child activity is needed.
215988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay     */
216988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay    public static Uri getDefaultRootUri(Activity activity) {
217988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay        return shouldShowDocumentsRoot(activity, activity.getIntent())
218988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay                ? DocumentsContract.buildHomeUri()
219988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay                : DocumentsContract.buildRootUri(
220988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay                        "com.android.providers.downloads.documents", "downloads");
221988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay    }
222988d8a354b00dce7e24deee187c08a4591956ac9Steve McKay
223741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska    public static boolean isHardwareKeyboardAvailable(Context context) {
224741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska        return context.getResources().getConfiguration().keyboard != Configuration.KEYBOARD_NOKEYS;
225741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska    }
226741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska
227741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska    public static void ensureKeyboardPresent(Context context, AlertDialog dialog) {
228741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska        if (!isHardwareKeyboardAvailable(context)) {
229741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska            dialog.getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_VISIBLE);
230741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska        }
231741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska    }
232741ac6fffb3ee2441569412a28a1c4f3c2160d87Aga Wronska
23364ae1f4e0236cf9ff06e19887c491bf08ee8adceAga Wronska    /*
23417f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     * Returns true if app is running in "productivity mode".
23564ae1f4e0236cf9ff06e19887c491bf08ee8adceAga Wronska     */
23617f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    private static boolean isProductivityMode(Context context, Intent intent) {
23717f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay        return intent.getBooleanExtra(
23817f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay                Shared.EXTRA_PRODUCTIVITY_MODE,
23917f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay                context.getResources().getBoolean(R.bool.productivity_device));
24017f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    }
24117f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay
24217f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    /*
24317f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     * Returns true if "Documents" root should be shown.
24417f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     */
24517f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    public static boolean shouldShowDocumentsRoot(Context context, Intent intent) {
24617f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay        return isProductivityMode(context, intent);
24717f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    }
24817f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay
24917f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    /*
25017f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     * Returns true if device root should be shown.
25117f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay     */
25217f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    public static boolean shouldShowDeviceRoot(Context context, Intent intent) {
25317f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay        return isProductivityMode(context, intent)
25417f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay                || intent.getBooleanExtra(DocumentsContract.EXTRA_SHOW_ADVANCED, false);
25517f7e5891cbe86fc6cacae045a4edd3e41f8d5eeSteve McKay    }
25624917427f41f9e41925c4a56d0c71d85e30643dfSteve McKay
25724917427f41f9e41925c4a56d0c71d85e30643dfSteve McKay    /**
25824917427f41f9e41925c4a56d0c71d85e30643dfSteve McKay     * Returns true if device root should be shown.
25924917427f41f9e41925c4a56d0c71d85e30643dfSteve McKay     */
26024917427f41f9e41925c4a56d0c71d85e30643dfSteve McKay    public static boolean shouldShowFancyFeatures(Activity activity) {
26124917427f41f9e41925c4a56d0c71d85e30643dfSteve McKay        Intent intent = activity.getIntent();
26224917427f41f9e41925c4a56d0c71d85e30643dfSteve McKay        return isProductivityMode(activity, intent)
26324917427f41f9e41925c4a56d0c71d85e30643dfSteve McKay                || intent.getBooleanExtra(DocumentsContract.EXTRA_FANCY_FEATURES, false);
26424917427f41f9e41925c4a56d0c71d85e30643dfSteve McKay    }
2652056205cb85d529d732638c45fe459168168e1eeGarfield, Tan
2662056205cb85d529d732638c45fe459168168e1eeGarfield, Tan    public static void checkMainLoop() {
2672056205cb85d529d732638c45fe459168168e1eeGarfield, Tan        if (Looper.getMainLooper() != Looper.myLooper()) {
2682056205cb85d529d732638c45fe459168168e1eeGarfield, Tan            Log.e(TAG, "Calling from non-UI thread!");
2692056205cb85d529d732638c45fe459168168e1eeGarfield, Tan        }
2702056205cb85d529d732638c45fe459168168e1eeGarfield, Tan    }
27185ec0d676297724f211213c7cb188839f1d3601bSteve McKay
27285ec0d676297724f211213c7cb188839f1d3601bSteve McKay    public static @Nullable <T> T findView(Activity activity, int... resources) {
27385ec0d676297724f211213c7cb188839f1d3601bSteve McKay        for (int id : resources) {
27485ec0d676297724f211213c7cb188839f1d3601bSteve McKay            @SuppressWarnings("unchecked")
27585ec0d676297724f211213c7cb188839f1d3601bSteve McKay            T r = (T) activity.findViewById(id);
27685ec0d676297724f211213c7cb188839f1d3601bSteve McKay            if (r != null) {
27785ec0d676297724f211213c7cb188839f1d3601bSteve McKay                return r;
27885ec0d676297724f211213c7cb188839f1d3601bSteve McKay            }
27985ec0d676297724f211213c7cb188839f1d3601bSteve McKay        }
28085ec0d676297724f211213c7cb188839f1d3601bSteve McKay        return null;
28185ec0d676297724f211213c7cb188839f1d3601bSteve McKay    }
282fefcd700d6b4cf1c4402af74c50fb0e762472901Steve McKay}
283