有些时候想自己写html网页,然后实现在我们的activity中与html相互操作,这种操作很少在app中提现,但是特殊情况下也是可以,哈哈。

看以下步骤:

首先需要写一个MainActivity的布局,我这里就简单的写一个,代码如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context="com.lai.jsdemo.MainActivity">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Android" />

    <Button
        android:id="@+id/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="8dp"
        android:background="@color/colorAccent"
        android:text="调用JS方法"
        android:textColor="#ffffff" />

    <Button
        android:id="@+id/button2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="8dp"
        android:background="@color/colorAccent"
        android:text="调用有参JS方法"
        android:textColor="#ffffff" />

    <TextView
        android:id="@+id/more"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="right"
        android:layout_marginRight="10dp"
        android:gravity="right"
        android:text="更多>>" />

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="8dp"
        android:text="WebView" />

    <WebView

        android:id="@+id/webview"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</LinearLayout>

一些简单的基本控件大家都知道了,需要注意的是这里用了一个WebView。

然后看下MainActivity类:

package com.lai.jsdemo;


import android.app.Activity;
import android.content.Intent;
import android.support.v7.app.AlertDialog;
import android.os.Bundle;
import android.view.View;
import android.webkit.JavascriptInterface;
import android.webkit.WebView;
import android.widget.TextView;
import android.widget.Toast;


public class MainActivity extends Activity {
    private WebView contentWebView = null;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        contentWebView = (WebView) findViewById(R.id.webview);
        TextView more= (TextView) findViewById(R.id.more);
        // 启用javascript
        contentWebView.getSettings().setJavaScriptEnabled(true);
        // 从assets目录下面的加载html
        contentWebView.loadUrl("file:///android_asset/web.html");
        contentWebView.addJavascriptInterface(MainActivity.this,"android");

        //无参调用Js点击
        findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                // 无参数调用
                contentWebView.loadUrl("javascript:javacalljs()");
            }
        });
        //有参调用Js点击
        findViewById(R.id.button2).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                // 传递参数调用
                contentWebView.loadUrl("javascript:javacalljswith(" + "''" + ")");
            }
        });
        //更多文本的点击事件
        more.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent moreIntent=new Intent(MainActivity.this,UserInfoActivity.class);
                startActivity(moreIntent);
                MainActivity.this.finish();
            }
        });
    }

    //由于安全原因 需要加 @JavascriptInterface
    @JavascriptInterface
    public void startFunction(){

        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                Toast.makeText(MainActivity.this,"MainActivity",Toast.LENGTH_SHORT).show();

            }
        });
    }

    @JavascriptInterface
    public void startFunction(final String text){
        runOnUiThread(new Runnable() {

            @Override
            public void run() {
               new AlertDialog.Builder(MainActivity.this).setMessage(text).show();

            }
        });


    }
}

到了这里相信大家多少有些疑问,不急,下面一一解答。

  • contentWebView.getSettings().setJavaScriptEnabled(true);
    如果访问的html页面中有 Javascript,则 webview 必须设置支持 Javascript。 webview.getSettings().setJavaScriptEnabled(true);
  • contentWebView.loadUrl(“file:///android_asset/web.html”);
    加载本地项目中assets目录下的web.html文件,注意正确写法。
  • contentWebView.addJavascriptInterface(MainActivity.this,”android”);
    addJavaScriptInterface是一个非常有用的功能,帮助我们从一个网页传递值到Android XML视图,后面你会看到在html中的点击会用得上“android”,学了html+css,JavaScript的应该知道window是JavaScript 层级中的顶层对象,在body中可以操作的东西可想而知,这里不说这些。
  • contentWebView.loadUrl(“javascript:javacalljs()”);
    调用JavaScript的方法,后面javacalljs()是function的方法。
  • 后面两个方法startFunction()是供给html中操作的,细心的朋友会注意到陌生的@JavascriptInterface,这边作重点讲一下。

因为安全问题,在Activity中给html操作的方法前面必须加上@JavascriptInterface注释,如果在targetSdkVersion在17之前该方法就能执行成功,如果在17之后(包括17)就不成功。


如果您在编写HTML5应用,需要在JS代码中访问Java中的函数,则你会用到WebView的addJavascriptInterface()函数。因为安全问题,在Android4.2中(如果应用的android:targetSdkVersion数值为17+)JS只能访问带有@JavascriptInterface注解的Java方法。 之前,任何public的函数都可以在JS代码中访问,而Java对象继承关系会导致很多public的方法都可以在JS中访问,其中一个重要的函数就是 getClass()。然后JS可以通过反射来访问其他一些内容。通过引入 @JavascriptInterface注解,则在JS中只能访问 @JavascriptInterface注解的函数。这样就可以增强安全性。

如果您的应用android:targetSdkVersion数值为17或者大于17记得添加 @JavascriptInterface 注解。


html网页的代码就不贴出来了,里面写的很简单,下面我会demo地址贴出来。

补充一点:
targetSdkVersion:targetSdkVersion 是设置希望的SDK版本,如果设置了此属性,那么在程序执行时,如果目标设备的API版本正好等于此数值,他会告诉Android平台:此程序在此版本已经经过充分测,没有问题。不必为此程序开启兼容性检查判断的工作了。也就是说,如果targetSdkVersion与目标设备的API版本相同时,运行效率可能会高一些。
但是,这个设置仅仅是一个声明、一个通知,不会有太实质的作用,
比如说,使用了targetSdkVersion这个SDK版本中的一个特性,但是这个特性在低版本中是不支持的,那么在低版本的API设备上运行程序时,可能会报错:java.lang.VerifyError。也就是说,此属性不会帮你解决兼容性的测试问题。

最后附上今天的demo:下载地址