| /* |
| * Copyright (C) 2010 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.browser; |
| |
| import android.app.AlertDialog; |
| import android.content.Context; |
| import android.content.DialogInterface; |
| import android.content.res.Configuration; |
| import android.net.http.SslCertificate; |
| import android.net.http.SslError; |
| import android.view.LayoutInflater; |
| import android.view.View; |
| import android.webkit.HttpAuthHandler; |
| import android.webkit.SslErrorHandler; |
| import android.webkit.WebView; |
| import android.webkit.WebViewClassic; |
| import android.widget.LinearLayout; |
| import android.widget.TextView; |
| |
| /** |
| * Displays page info |
| * |
| */ |
| public class PageDialogsHandler { |
| |
| private Context mContext; |
| private Controller mController; |
| private boolean mPageInfoFromShowSSLCertificateOnError; |
| private String mUrlCertificateOnError; |
| private Tab mPageInfoView; |
| private AlertDialog mPageInfoDialog; |
| |
| // as SSLCertificateOnError has different style for landscape / portrait, |
| // we have to re-open it when configuration changed |
| private AlertDialog mSSLCertificateOnErrorDialog; |
| private WebView mSSLCertificateOnErrorView; |
| private SslErrorHandler mSSLCertificateOnErrorHandler; |
| private SslError mSSLCertificateOnErrorError; |
| |
| // as SSLCertificate has different style for landscape / portrait, we |
| // have to re-open it when configuration changed |
| private AlertDialog mSSLCertificateDialog; |
| private Tab mSSLCertificateView; |
| private HttpAuthenticationDialog mHttpAuthenticationDialog; |
| |
| public PageDialogsHandler(Context context, Controller controller) { |
| mContext = context; |
| mController = controller; |
| } |
| |
| public void onConfigurationChanged(Configuration config) { |
| if (mPageInfoDialog != null) { |
| mPageInfoDialog.dismiss(); |
| showPageInfo(mPageInfoView, |
| mPageInfoFromShowSSLCertificateOnError, |
| mUrlCertificateOnError); |
| } |
| if (mSSLCertificateDialog != null) { |
| mSSLCertificateDialog.dismiss(); |
| showSSLCertificate(mSSLCertificateView); |
| } |
| if (mSSLCertificateOnErrorDialog != null) { |
| mSSLCertificateOnErrorDialog.dismiss(); |
| showSSLCertificateOnError(mSSLCertificateOnErrorView, |
| mSSLCertificateOnErrorHandler, |
| mSSLCertificateOnErrorError); |
| } |
| if (mHttpAuthenticationDialog != null) { |
| mHttpAuthenticationDialog.reshow(); |
| } |
| } |
| |
| /** |
| * Displays an http-authentication dialog. |
| */ |
| void showHttpAuthentication(final Tab tab, final HttpAuthHandler handler, String host, String realm) { |
| mHttpAuthenticationDialog = new HttpAuthenticationDialog(mContext, host, realm); |
| mHttpAuthenticationDialog.setOkListener(new HttpAuthenticationDialog.OkListener() { |
| public void onOk(String host, String realm, String username, String password) { |
| setHttpAuthUsernamePassword(host, realm, username, password); |
| handler.proceed(username, password); |
| mHttpAuthenticationDialog = null; |
| } |
| }); |
| mHttpAuthenticationDialog.setCancelListener(new HttpAuthenticationDialog.CancelListener() { |
| public void onCancel() { |
| handler.cancel(); |
| mController.onUpdatedSecurityState(tab); |
| mHttpAuthenticationDialog = null; |
| } |
| }); |
| mHttpAuthenticationDialog.show(); |
| } |
| |
| /** |
| * Set HTTP authentication password. |
| * |
| * @param host The host for the password |
| * @param realm The realm for the password |
| * @param username The username for the password. If it is null, it means |
| * password can't be saved. |
| * @param password The password |
| */ |
| public void setHttpAuthUsernamePassword(String host, String realm, |
| String username, |
| String password) { |
| WebView w = mController.getCurrentTopWebView(); |
| if (w != null) { |
| w.setHttpAuthUsernamePassword(host, realm, username, password); |
| } |
| } |
| |
| /** |
| * Displays a page-info dialog. |
| * @param tab The tab to show info about |
| * @param fromShowSSLCertificateOnError The flag that indicates whether |
| * this dialog was opened from the SSL-certificate-on-error dialog or |
| * not. This is important, since we need to know whether to return to |
| * the parent dialog or simply dismiss. |
| * @param urlCertificateOnError The URL that invokes SSLCertificateError. |
| * Null when fromShowSSLCertificateOnError is false. |
| */ |
| void showPageInfo(final Tab tab, |
| final boolean fromShowSSLCertificateOnError, |
| final String urlCertificateOnError) { |
| if (tab == null) return; |
| final LayoutInflater factory = LayoutInflater.from(mContext); |
| |
| final View pageInfoView = factory.inflate(R.layout.page_info, null); |
| |
| final WebView view = tab.getWebView(); |
| |
| String url = fromShowSSLCertificateOnError ? urlCertificateOnError : tab.getUrl(); |
| String title = tab.getTitle(); |
| |
| if (url == null) { |
| url = ""; |
| } |
| if (title == null) { |
| title = ""; |
| } |
| |
| ((TextView) pageInfoView.findViewById(R.id.address)).setText(url); |
| ((TextView) pageInfoView.findViewById(R.id.title)).setText(title); |
| |
| mPageInfoView = tab; |
| mPageInfoFromShowSSLCertificateOnError = fromShowSSLCertificateOnError; |
| mUrlCertificateOnError = urlCertificateOnError; |
| |
| AlertDialog.Builder alertDialogBuilder = |
| new AlertDialog.Builder(mContext) |
| .setTitle(R.string.page_info) |
| .setIcon(android.R.drawable.ic_dialog_info) |
| .setView(pageInfoView) |
| .setPositiveButton( |
| R.string.ok, |
| new DialogInterface.OnClickListener() { |
| public void onClick(DialogInterface dialog, |
| int whichButton) { |
| mPageInfoDialog = null; |
| mPageInfoView = null; |
| |
| // if we came here from the SSL error dialog |
| if (fromShowSSLCertificateOnError) { |
| // go back to the SSL error dialog |
| showSSLCertificateOnError( |
| mSSLCertificateOnErrorView, |
| mSSLCertificateOnErrorHandler, |
| mSSLCertificateOnErrorError); |
| } |
| } |
| }) |
| .setOnCancelListener( |
| new DialogInterface.OnCancelListener() { |
| public void onCancel(DialogInterface dialog) { |
| mPageInfoDialog = null; |
| mPageInfoView = null; |
| |
| // if we came here from the SSL error dialog |
| if (fromShowSSLCertificateOnError) { |
| // go back to the SSL error dialog |
| showSSLCertificateOnError( |
| mSSLCertificateOnErrorView, |
| mSSLCertificateOnErrorHandler, |
| mSSLCertificateOnErrorError); |
| } |
| } |
| }); |
| |
| // if we have a main top-level page SSL certificate set or a certificate |
| // error |
| if (fromShowSSLCertificateOnError || |
| (view != null && view.getCertificate() != null)) { |
| // add a 'View Certificate' button |
| alertDialogBuilder.setNeutralButton( |
| R.string.view_certificate, |
| new DialogInterface.OnClickListener() { |
| public void onClick(DialogInterface dialog, |
| int whichButton) { |
| mPageInfoDialog = null; |
| mPageInfoView = null; |
| |
| // if we came here from the SSL error dialog |
| if (fromShowSSLCertificateOnError) { |
| // go back to the SSL error dialog |
| showSSLCertificateOnError( |
| mSSLCertificateOnErrorView, |
| mSSLCertificateOnErrorHandler, |
| mSSLCertificateOnErrorError); |
| } else { |
| // otherwise, display the top-most certificate from |
| // the chain |
| showSSLCertificate(tab); |
| } |
| } |
| }); |
| } |
| |
| mPageInfoDialog = alertDialogBuilder.show(); |
| } |
| |
| /** |
| * Displays the main top-level page SSL certificate dialog |
| * (accessible from the Page-Info dialog). |
| * @param tab The tab to show certificate for. |
| */ |
| private void showSSLCertificate(final Tab tab) { |
| |
| SslCertificate cert = tab.getWebView().getCertificate(); |
| if (cert == null) { |
| return; |
| } |
| |
| mSSLCertificateView = tab; |
| mSSLCertificateDialog = createSslCertificateDialog(cert, tab.getSslCertificateError()) |
| .setPositiveButton(R.string.ok, |
| new DialogInterface.OnClickListener() { |
| public void onClick(DialogInterface dialog, |
| int whichButton) { |
| mSSLCertificateDialog = null; |
| mSSLCertificateView = null; |
| |
| showPageInfo(tab, false, null); |
| } |
| }) |
| .setOnCancelListener( |
| new DialogInterface.OnCancelListener() { |
| public void onCancel(DialogInterface dialog) { |
| mSSLCertificateDialog = null; |
| mSSLCertificateView = null; |
| |
| showPageInfo(tab, false, null); |
| } |
| }) |
| .show(); |
| } |
| |
| /** |
| * Displays the SSL error certificate dialog. |
| * @param view The target web-view. |
| * @param handler The SSL error handler responsible for cancelling the |
| * connection that resulted in an SSL error or proceeding per user request. |
| * @param error The SSL error object. |
| */ |
| void showSSLCertificateOnError( |
| final WebView view, final SslErrorHandler handler, |
| final SslError error) { |
| |
| SslCertificate cert = error.getCertificate(); |
| if (cert == null) { |
| return; |
| } |
| |
| mSSLCertificateOnErrorHandler = handler; |
| mSSLCertificateOnErrorView = view; |
| mSSLCertificateOnErrorError = error; |
| mSSLCertificateOnErrorDialog = createSslCertificateDialog(cert, error) |
| .setPositiveButton(R.string.ok, |
| new DialogInterface.OnClickListener() { |
| public void onClick(DialogInterface dialog, |
| int whichButton) { |
| mSSLCertificateOnErrorDialog = null; |
| mSSLCertificateOnErrorView = null; |
| mSSLCertificateOnErrorHandler = null; |
| mSSLCertificateOnErrorError = null; |
| |
| WebViewClassic.fromWebView(view).getWebViewClient(). |
| onReceivedSslError(view, handler, error); |
| } |
| }) |
| .setNeutralButton(R.string.page_info_view, |
| new DialogInterface.OnClickListener() { |
| public void onClick(DialogInterface dialog, |
| int whichButton) { |
| mSSLCertificateOnErrorDialog = null; |
| |
| // do not clear the dialog state: we will |
| // need to show the dialog again once the |
| // user is done exploring the page-info details |
| |
| showPageInfo(mController.getTabControl() |
| .getTabFromView(view), |
| true, |
| error.getUrl()); |
| } |
| }) |
| .setOnCancelListener( |
| new DialogInterface.OnCancelListener() { |
| public void onCancel(DialogInterface dialog) { |
| mSSLCertificateOnErrorDialog = null; |
| mSSLCertificateOnErrorView = null; |
| mSSLCertificateOnErrorHandler = null; |
| mSSLCertificateOnErrorError = null; |
| |
| WebViewClassic.fromWebView(view).getWebViewClient(). |
| onReceivedSslError(view, handler, error); |
| } |
| }) |
| .show(); |
| } |
| |
| /* |
| * Creates an AlertDialog to display the given certificate. If error is |
| * null, text is added to state that the certificae is valid and the icon |
| * is set accordingly. If error is non-null, it must relate to the supplied |
| * certificate. In this case, error is used to add text describing the |
| * problems with the certificate and a different icon is used. |
| */ |
| private AlertDialog.Builder createSslCertificateDialog(SslCertificate certificate, |
| SslError error) { |
| View certificateView = certificate.inflateCertificateView(mContext); |
| final LinearLayout placeholder = |
| (LinearLayout)certificateView.findViewById(com.android.internal.R.id.placeholder); |
| |
| LayoutInflater factory = LayoutInflater.from(mContext); |
| int iconId; |
| |
| if (error == null) { |
| iconId = R.drawable.ic_dialog_browser_certificate_secure; |
| LinearLayout table = (LinearLayout)factory.inflate(R.layout.ssl_success, placeholder); |
| TextView successString = (TextView)table.findViewById(R.id.success); |
| successString.setText(com.android.internal.R.string.ssl_certificate_is_valid); |
| } else { |
| iconId = R.drawable.ic_dialog_browser_certificate_partially_secure; |
| if (error.hasError(SslError.SSL_UNTRUSTED)) { |
| addError(factory, placeholder, R.string.ssl_untrusted); |
| } |
| if (error.hasError(SslError.SSL_IDMISMATCH)) { |
| addError(factory, placeholder, R.string.ssl_mismatch); |
| } |
| if (error.hasError(SslError.SSL_EXPIRED)) { |
| addError(factory, placeholder, R.string.ssl_expired); |
| } |
| if (error.hasError(SslError.SSL_NOTYETVALID)) { |
| addError(factory, placeholder, R.string.ssl_not_yet_valid); |
| } |
| if (error.hasError(SslError.SSL_DATE_INVALID)) { |
| addError(factory, placeholder, R.string.ssl_date_invalid); |
| } |
| if (error.hasError(SslError.SSL_INVALID)) { |
| addError(factory, placeholder, R.string.ssl_invalid); |
| } |
| // The SslError should always have at least one type of error and we |
| // should explicitly handle every type of error it supports. We |
| // therefore expect the condition below to never be hit. We use it |
| // as as safety net in case a new error type is added to SslError |
| // without the logic above being updated accordingly. |
| if (placeholder.getChildCount() == 0) { |
| addError(factory, placeholder, R.string.ssl_unknown); |
| } |
| } |
| |
| return new AlertDialog.Builder(mContext) |
| .setTitle(com.android.internal.R.string.ssl_certificate) |
| .setIcon(iconId) |
| .setView(certificateView); |
| } |
| |
| private void addError(LayoutInflater inflater, LinearLayout parent, int error) { |
| TextView textView = (TextView) inflater.inflate(R.layout.ssl_warning, |
| parent, false); |
| textView.setText(error); |
| parent.addView(textView); |
| } |
| } |