/* * Copyright (C) 2011 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 com.android.vpndialogs; import static android.view.WindowManager.LayoutParams.SYSTEM_FLAG_HIDE_NON_SYSTEM_OVERLAY_WINDOWS; import android.content.DialogInterface; import android.content.pm.PackageManager; import android.graphics.drawable.Drawable; import android.net.VpnManager; import android.os.Bundle; import android.os.UserHandle; import android.os.UserManager; import android.text.Html; import android.text.Html.ImageGetter; import android.util.Log; import android.util.TypedValue; import android.view.View; import android.widget.Button; import android.widget.TextView; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.app.AlertActivity; import com.android.internal.net.VpnConfig; public class ConfirmDialog extends AlertActivity implements DialogInterface.OnClickListener, ImageGetter { private static final String TAG = "VpnConfirm"; // Usually the label represents the app name, 150 code points might be enough to display the app // name, and 150 code points won't cover the warning message from VpnDialog. @VisibleForTesting static final int MAX_VPN_LABEL_LENGTH = 150; @VpnManager.VpnType private final int mVpnType; private String mPackage; private VpnManager mVm; private View mView; public ConfirmDialog() { this(VpnManager.TYPE_VPN_SERVICE); } public ConfirmDialog(@VpnManager.VpnType int vpnType) { mVpnType = vpnType; } /** * This function will use the string resource to combine the VPN label and the package name. * * If the VPN label violates the length restriction, the first 30 code points of VPN label and * the package name will be returned. Or return the VPN label and the package name directly if * the VPN label doesn't violate the length restriction. * * The result will be something like, * - ThisIsAVeryLongVpnAppNameWhich... (com.vpn.app) * if the VPN label violates the length restriction. * or * - VpnLabelWith<br>HtmlTag (com.vpn.app) * if the VPN label doesn't violate the length restriction. * */ private String getSimplifiedLabel(String vpnLabel, String packageName) { if (vpnLabel.codePointCount(0, vpnLabel.length()) > 30) { return getString(R.string.sanitized_vpn_label_with_ellipsis, vpnLabel.substring(0, vpnLabel.offsetByCodePoints(0, 30)), packageName); } return getString(R.string.sanitized_vpn_label, vpnLabel, packageName); } @VisibleForTesting protected String getSanitizedVpnLabel(String vpnLabel, String packageName) { final String sanitizedVpnLabel = Html.escapeHtml(vpnLabel); final boolean exceedMaxVpnLabelLength = sanitizedVpnLabel.codePointCount(0, sanitizedVpnLabel.length()) > MAX_VPN_LABEL_LENGTH; if (exceedMaxVpnLabelLength || !vpnLabel.equals(sanitizedVpnLabel)) { return getSimplifiedLabel(sanitizedVpnLabel, packageName); } return sanitizedVpnLabel; } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mPackage = getCallingPackage(); mVm = getSystemService(VpnManager.class); if (mVm.prepareVpn(mPackage, null, UserHandle.myUserId())) { setResult(RESULT_OK); finish(); return; } if (UserManager.get(this).hasUserRestriction(UserManager.DISALLOW_CONFIG_VPN)) { finish(); return; } final String alwaysOnVpnPackage = mVm.getAlwaysOnVpnPackageForUser(UserHandle.myUserId()); // Can't prepare new vpn app when another vpn is always-on if (alwaysOnVpnPackage != null && !alwaysOnVpnPackage.equals(mPackage)) { finish(); return; } mView = View.inflate(this, R.layout.confirm, null); ((TextView) mView.findViewById(R.id.warning)).setText( Html.fromHtml(getString(R.string.warning, getSanitizedVpnLabel( getVpnLabel().toString(), mPackage)), this /* imageGetter */, null /* tagHandler */)); mAlertParams.mTitle = getText(R.string.prompt); mAlertParams.mPositiveButtonText = getText(android.R.string.ok); mAlertParams.mPositiveButtonListener = this; mAlertParams.mNegativeButtonText = getText(android.R.string.cancel); mAlertParams.mView = mView; setupAlert(); getWindow().setCloseOnTouchOutside(false); getWindow().addPrivateFlags(SYSTEM_FLAG_HIDE_NON_SYSTEM_OVERLAY_WINDOWS); Button button = mAlert.getButton(DialogInterface.BUTTON_POSITIVE); button.setFilterTouchesWhenObscured(true); } @VisibleForTesting public CharSequence getWarningText() { return ((TextView) mView.findViewById(R.id.warning)).getText(); } private CharSequence getVpnLabel() { try { return VpnConfig.getVpnLabel(this, mPackage); } catch (PackageManager.NameNotFoundException e) { throw new IllegalStateException(e); } } @Override public Drawable getDrawable(String source) { // Should only reach this when fetching the VPN icon for the warning string. final Drawable icon = getDrawable(R.drawable.ic_vpn_dialog); icon.setBounds(0, 0, icon.getIntrinsicWidth(), icon.getIntrinsicHeight()); final TypedValue tv = new TypedValue(); if (getTheme().resolveAttribute(android.R.attr.textColorPrimary, tv, true)) { icon.setTint(getColor(tv.resourceId)); } else { Log.w(TAG, "Unable to resolve theme color"); } return icon; } @Override public void onBackPressed() { } @Override public void onClick(DialogInterface dialog, int which) { try { if (mVm.prepareVpn(null, mPackage, UserHandle.myUserId())) { // Authorize this app to initiate VPN connections in the future without user // intervention. mVm.setVpnPackageAuthorization(mPackage, UserHandle.myUserId(), mVpnType); setResult(RESULT_OK); } } catch (Exception e) { Log.e(TAG, "onClick", e); } } }