Webview电子邮件链接(mailto)

39

我有一个视图,并且该网站有恶意代码来发送电子邮件。 当我打开链接时会出现错误。 我希望当我打开链接时,能够打开Gmail应用程序或其他电子邮件应用程序。 感谢所有的帮助者。

public class teacher extends Activity implements OnClickListener {
    WebView webView;
    final Activity activity = this;    
    @Override
    public void onCreate(Bundle savedInstanceState) {    
        super.onCreate(savedInstanceState);
        this.getWindow().requestFeature(Window.FEATURE_PROGRESS);
        setContentView(R.layout.teacher);
        webView = (WebView) findViewById(R.id.webView145);    
        webView.getSettings().setJavaScriptEnabled(true);           webView.loadUrl("https://dl.dropboxusercontent.com/u/233211/%D7%A8%D7%A9%D7%99%D7%95%D7%9F/iWebKit%20demo/ther.html");    
        webView.setWebChromeClient(new WebChromeClient() {
            public void onProgressChanged(WebView view, int progress) {
                activity.setTitle("Loading...");
                activity.setProgress(progress * 100);    
                if (progress == 100)
                activity.setTitle(R.string.app_name); 
            }       });    
        webView.setWebViewClient(new WebViewClient() {
            @Override
            public void onReceivedError(WebView view, int errorCode,
            String description, String failingUrl) {
            // Handle the error
            }
            @Override
            public boolean shouldOverrideUrlLoading(WebView view, String url) {
                view.loadUrl(url);
                return true;            }           });}
        public void onBackPressed() {    
        if (webView.canGoBack()) {
            webView.goBack();
        } else {
            Intent B = new Intent(this, MainActivity.class);
            startActivity(B);
        }}    
    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        if (event.getAction() == KeyEvent.ACTION_DOWN) {
            switch (keyCode) {
            case KeyEvent.KEYCODE_BACK:
                if (webView.canGoBack() == true) {
                    webView.goBack();
                } else {
                    finish();
            }return true;}          }
        return super.onKeyDown(keyCode, event);
    }
    @Override
    public void onClick(View arg0) {
        // TODO Auto-generated method stub

    }}

请描述“出现错误”的情况。 - CommonsWare
3个回答

56
你需要创建一个WebViewClient的子类并重写mailto URL加载。例如:
public class MyWebViewClient extends WebViewClient {
  private final WeakReference<Activity> mActivityRef;

  public MyWebViewClient(Activity activity) {
    mActivityRef = new WeakReference<Activity>(activity);
  }

  @Override
  public boolean shouldOverrideUrlLoading(WebView view, String url) {
    if (url.startsWith("mailto:")) {
      final Activity activity = mActivityRef.get();
      if (activity != null) {
        MailTo mt = MailTo.parse(url);
        Intent i = newEmailIntent(activity, mt.getTo(), mt.getSubject(), mt.getBody(), mt.getCc());
        activity.startActivity(i);
        view.reload();
        return true;
      }
    } else {
      view.loadUrl(url);
    }
    return true;
  }

  private Intent newEmailIntent(Context context, String address, String subject, String body, String cc) {
    Intent intent = new Intent(Intent.ACTION_SEND);
    intent.putExtra(Intent.EXTRA_EMAIL, new String[] { address });
    intent.putExtra(Intent.EXTRA_TEXT, body);
    intent.putExtra(Intent.EXTRA_SUBJECT, subject);
    intent.putExtra(Intent.EXTRA_CC, cc);
    intent.setType("message/rfc822");
    return intent;
  }
}

那么您需要将这个自定义的WebViewClient设置为您的WebView:

webView.setWebViewClient(new MyWebViewClient(activity);

我无法将“webView.setWebViewClient(new WestwingWebViewClient(activity);”添加到我的webview中。因为无法解析westwingWebViewClient。 - omer341
更新的答案。您应该设置您新创建的自定义WebViewClient,它是MyWebViewClient。 - Tomas Žemaitis
4
为什么在 else 中会有 view.loadUrl(url);?是否可以在 if 后面简单地使用 return false; 而不需要 else 呢? - Glenn85
2
我建议将 "newEmailIntent" 的最后一行替换为 return Intent.createChooser(intent, "选择器的标题"); - Quentin G.

38
你应该使用以下内容更新你的WebViewClient:
@SuppressWarnings("deprecation") 
@Override 
public boolean shouldOverrideUrlLoading(WebView view, String url) {
    proceedUrl(view, Uri.parse(url))
    return true; 
} 

@TargetApi(Build.VERSION_CODES.N)
@Override 
public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) {
    proceedUrl(view, request.getUrl());
    return true; 
}

private void proceedUrl(View view, Uri uri){  
    if (uri.toString().startsWith("mailto:")) {
        startActivity(new Intent(Intent.ACTION_SENDTO, uri));
    } else if (uri.toString().startsWith("tel:")) {
        startActivity(new Intent(Intent.ACTION_DIAL, uri));
    } else { 
        view.loadUrl(uri.toString());
    } 
} 

26

注意:- 在安卓Nougat之后,shouldOverrideUrlLoading已经废弃

您需要使用shouldInterceptRequestshouldOverrideUrlLoading来获得更好的支持。

此外,您可能想要检查URL中是否有mailto:tel:,这些在HTML5中分别用于触发邮件客户端和电话拨号。

现在完整的解决方案将看起来像这样

@SuppressWarnings("deprecation") 
@Override 
public boolean shouldOverrideUrlLoading(WebView view, String url) {
    if (url.startsWith("mailto:")) {  
        //Handle mail Urls 
        startActivity(new Intent(Intent.ACTION_SENDTO, Uri.parse(url)));
    } else if (url.startsWith("tel:")) {
        //Handle telephony Urls 
        startActivity(new Intent(Intent.ACTION_DIAL, Uri.parse(url)));
    } else { 
        view.loadUrl(url);
    } 
    return true; 
} 

@TargetApi(Build.VERSION_CODES.N)
@Override 
public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) {
    final Uri uri = request.getUrl();
    if (uri.toString().startsWith("mailto:")) {
        //Handle mail Urls 
        startActivity(new Intent(Intent.ACTION_SENDTO, uri));
    } else if (uri.toString().startsWith("tel:")) {
        //Handle telephony Urls 
        startActivity(new Intent(Intent.ACTION_DIAL, uri));
    } else { 
        //Handle Web Urls 
        view.loadUrl(uri.toString());
    } 
    return true; 
} 

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接