How Can I Check Whether Chrome Supports Chrome Custom Tabs?
Solution 1:
Instead of binding and unbinding the service, you can use the PackageManager to check if Custom Tabs is supported.
private static final String SERVICE_ACTION = "android.support.customtabs.action.CustomTabsService";
private static final String CHROME_PACKAGE = "com.android.chrome";
private static boolean isChromeCustomTabsSupported(@NonNull final Context context) {
Intent serviceIntent = new Intent(SERVICE_ACTION);
serviceIntent.setPackage(CHROME_PACKAGE);
List<ResolveInfo> resolveInfos = context.getPackageManager().queryIntentServices(serviceIntent, 0);
return !(resolveInfos == null || resolveInfos.isEmpty());
}
Be aware that other browsers may support Custom Tabs in the future, so you may want to modify that to support this case.
Solution 2:
You can try following code to figure out if you have a browser that supports custom tab:
private static final String TAG = "CustomTabLauncher";
static final String STABLE_PACKAGE = "com.android.chrome";
static final String BETA_PACKAGE = "com.chrome.beta";
static final String DEV_PACKAGE = "com.chrome.dev";
static final String LOCAL_PACKAGE = "com.google.android.apps.chrome";
String mPackageNameToUse;
private String getPackageName(Context context) {
if (mPackageNameToUse != null) {
return mPackageNameToUse;
}
// Get default VIEW intent handler that can view a web url.
Intent activityIntent = new Intent(Intent.ACTION_VIEW, Uri.parse("http://www.test-url.com"));
// Get all apps that can handle VIEW intents.
PackageManager pm = context.getPackageManager();
List<ResolveInfo> resolvedActivityList = pm.queryIntentActivities(activityIntent, 0);
List<String> packagesSupportingCustomTabs = new ArrayList<>();
for (ResolveInfo info : resolvedActivityList) {
Intent serviceIntent = new Intent();
serviceIntent.setAction(CustomTabsService.ACTION_CUSTOM_TABS_CONNECTION);
serviceIntent.setPackage(info.activityInfo.packageName);
if (pm.resolveService(serviceIntent, 0) != null) {
packagesSupportingCustomTabs.add(info.activityInfo.packageName);
}
}
// Now packagesSupportingCustomTabs contains all apps that can handle both VIEW intents
// and service calls.
if (packagesSupportingCustomTabs.isEmpty()) {
mPackageNameToUse = null;
} else if (packagesSupportingCustomTabs.size() == 1) {
mPackageNameToUse = packagesSupportingCustomTabs.get(0);
} else if (packagesSupportingCustomTabs.contains(STABLE_PACKAGE)) {
mPackageNameToUse = STABLE_PACKAGE;
} else if (packagesSupportingCustomTabs.contains(BETA_PACKAGE)) {
mPackageNameToUse = BETA_PACKAGE;
} else if (packagesSupportingCustomTabs.contains(DEV_PACKAGE)) {
mPackageNameToUse = DEV_PACKAGE;
} else if (packagesSupportingCustomTabs.contains(LOCAL_PACKAGE)) {
mPackageNameToUse = LOCAL_PACKAGE;
}
return mPackageNameToUse;
}
When calling, you can do something like this:
public void openCustomTab(Uri uri, Activity activity) {
//If we cant find a package name, it means there's no browser that supports
//Chrome Custom Tabs installed. So, we fallback to the default browser
if (getPackageName(activity) == null) {
activity.startActivity(new Intent(Intent.ACTION_VIEW, uri));
} else {
CustomTabsIntent.Builder intentBuilder = new CustomTabsIntent.Builder();
intentBuilder.enableUrlBarHiding();
intentBuilder.setToolbarColor(activity.getResources().getColor(R.color.purple_a_01));
CustomTabsIntent customTabsIntent = intentBuilder.build();
customTabsIntent.intent.setPackage(mPackageNameToUse);
customTabsIntent.launchUrl(activity, uri);
}
}
Solution 3:
I ended up writing a static method in my Utils class so I can check and handle the case where it isn't supported:
/**
* Check if Chrome CustomTabs are supported.
* Some devices don't have Chrome or it may not be
* updated to a version where custom tabs is supported.
*
* @param context the context
* @return whether custom tabs are supported
*/
public static boolean isChromeCustomTabsSupported(@NonNull final Context context) {
Intent serviceIntent = new Intent("android.support.customtabs.action.CustomTabsService");
serviceIntent.setPackage("com.android.chrome");
CustomTabsServiceConnection serviceConnection = new CustomTabsServiceConnection() {
@Override
public void onCustomTabsServiceConnected(final ComponentName componentName, final CustomTabsClient customTabsClient) { }
@Override
public void onServiceDisconnected(final ComponentName name) { }
};
boolean customTabsSupported =
context.bindService(serviceIntent, serviceConnection, Context.BIND_AUTO_CREATE | Context.BIND_WAIVE_PRIORITY);
context.unbindService(serviceConnection);
return customTabsSupported;
}
Solution 4:
I solved this problem by handling ActivityNotFound
exception in catch block.
The trick is to check if the browser activity of chrome can be started or not, if it can't be started or throws an exception then simply open the link through Intent.ACTION_VIEW
.
Here is all the relevant code ....
private void onBtnLinkClicked(View v, int pos) {
try {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
openCustomTab(url);
} else {
openBrowserActivity(url);
}
} catch (Exception e) {
e.printStackTrace();
openBrowserActivity(url);
}
}
private void openBrowserActivity(String url) {
Intent browserIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
context.startActivity(browserIntent);
}
What is openCustomTab(url)
you say :
Here is the relevant code for it.
private void openCustomTab(String url) {
CustomTabsIntent.Builder intentBuilder = new CustomTabsIntent.Builder();
int color = context.getResources().getColor(R.color.colorPrimary);
intentBuilder.setToolbarColor(color);
intentBuilder.setShowTitle(true);
String menuItemTitle = context.getString(R.string.menu_title_share);
PendingIntent menuItemPendingIntent = createPendingShareIntent();
intentBuilder.addMenuItem(menuItemTitle, menuItemPendingIntent);
intentBuilder.setStartAnimations(context,
R.anim.slide_in_right, R.anim.slide_out_left);
intentBuilder.setExitAnimations(context,
android.R.anim.slide_in_left, android.R.anim.slide_out_right);
CustomTabActivityHelper.openCustomTab(
activity, intentBuilder.build(), Uri.parse(url), new WebviewFallback());
}
My style of answer may seem cocky but before clicking downvote let me know if you have run into any unexpected bug or any other problem that this approach may have cause. Do give your feedback, we are a community afterall.
Solution 5:
New issues may arise if you are targeting API level 30 and running on an Android 11 device.
You will need to add a section to your manifest, such as:
<queries>
<intent>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.BROWSABLE" />
<data android:scheme="https" />
</intent>
</queries>
OR
<queries>
<intent>
<action android:name="android.support.customtabs.action.CustomTabsService" />
</intent>
</queries>
Without this, some of the calls to PackageManager mentioned in the above posts will not work as you expect.
https://developer.android.com/training/package-visibility/use-cases
Post a Comment for "How Can I Check Whether Chrome Supports Chrome Custom Tabs?"