有些时候想自己写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:下载地址