Android:use webview to execute js

This commit is contained in:
pengfei.zhou 2021-11-05 15:14:32 +08:00 committed by osborn
parent 196497f3bd
commit bb17b74b99
7 changed files with 367 additions and 42 deletions

View File

@ -97,6 +97,7 @@ public class DoricJSEngine implements Handler.Callback, DoricTimerExtension.Time
protected void initJSEngine() { protected void initJSEngine() {
mDoricJSE = new DoricWebViewJSExecutor(Doric.application()); mDoricJSE = new DoricWebViewJSExecutor(Doric.application());
loadBuiltinJS("doric-web.js");
} }
public void setEnvironmentValue(Map<String, Object> values) { public void setEnvironmentValue(Map<String, Object> values) {
@ -239,9 +240,6 @@ public class DoricJSEngine implements Handler.Callback, DoricTimerExtension.Time
private void initDoricRuntime() { private void initDoricRuntime() {
try { try {
if (mDoricJSE instanceof DoricWebViewJSExecutor) {
loadBuiltinJS("doric-web.js");
}
loadBuiltinJS(DoricConstant.DORIC_BUNDLE_SANDBOX); loadBuiltinJS(DoricConstant.DORIC_BUNDLE_SANDBOX);
String libName = DoricConstant.DORIC_MODULE_LIB; String libName = DoricConstant.DORIC_MODULE_LIB;
String libJS = DoricUtils.readAssetFile(DoricConstant.DORIC_BUNDLE_LIB); String libJS = DoricUtils.readAssetFile(DoricConstant.DORIC_BUNDLE_LIB);

View File

@ -17,6 +17,7 @@ package pub.doric.engine;
import android.annotation.SuppressLint; import android.annotation.SuppressLint;
import android.content.Context; import android.content.Context;
import android.text.TextUtils;
import android.webkit.ConsoleMessage; import android.webkit.ConsoleMessage;
import android.webkit.JavascriptInterface; import android.webkit.JavascriptInterface;
import android.webkit.JsResult; import android.webkit.JsResult;
@ -37,6 +38,7 @@ import org.json.JSONObject;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
import pub.doric.async.SettableFuture;
import pub.doric.utils.DoricLog; import pub.doric.utils.DoricLog;
@ -59,22 +61,83 @@ public class DoricWebViewJSExecutor implements IDoricJSE {
case "boolean": case "boolean":
return jsonObject.optBoolean("value"); return jsonObject.optBoolean("value");
case "object": case "object":
return jsonObject.optJSONObject("value"); try {
return new JSONObject(jsonObject.optString("value"));
} catch (JSONException e) {
e.printStackTrace();
return JSONObject.NULL;
}
case "array": case "array":
return jsonObject.optJSONArray("value"); try {
return new JSONArray(jsonObject.optString("value"));
} catch (JSONException e) {
e.printStackTrace();
return JSONObject.NULL;
}
default: default:
return JSONObject.NULL; return JSONObject.NULL;
} }
} }
private static final String WRAPPED_NULL = new JSONBuilder().put("type", "null").toString(); private static JSONObject wrapJavaValue(JavaValue javaValue) {
if (javaValue == null || javaValue.getType() == 0) {
return WRAPPED_NULL;
}
if (javaValue.getType() == 1) {
Double value = Double.valueOf(javaValue.getValue());
return new JSONBuilder().put("type", "number").put("value", value).toJSONObject();
}
if (javaValue.getType() == 2) {
Boolean value = Boolean.valueOf(javaValue.getValue());
return new JSONBuilder().put("type", "boolean").put("value", value).toJSONObject();
}
if (javaValue.getType() == 3) {
String value = String.valueOf(javaValue.getValue());
return new JSONBuilder().put("type", "string").put("value", value).toJSONObject();
}
if (javaValue.getType() == 4) {
String value = String.valueOf(javaValue.getValue());
try {
return new JSONBuilder().put("type", "object").put("value", new JSONObject(value)).toJSONObject();
} catch (JSONException e) {
e.printStackTrace();
}
}
if (javaValue.getType() == 5) {
String value = String.valueOf(javaValue.getValue());
try {
return new JSONBuilder().put("type", "array").put("value", new JSONArray(value)).toJSONObject();
} catch (JSONException e) {
e.printStackTrace();
}
}
return WRAPPED_NULL;
}
private static final JSONObject WRAPPED_NULL = new JSONBuilder().put("type", "null").toJSONObject();
private static final String WRAPPED_NULL_STRING = WRAPPED_NULL.toString();
private SettableFuture<String> returnFuture = null;
public class WebViewCallback { public class WebViewCallback {
@JavascriptInterface
public void log(String message) {
DoricLog.d(message);
}
@JavascriptInterface
public void returnNative(String result) {
if (returnFuture != null) {
returnFuture.set(result);
}
}
@JavascriptInterface @JavascriptInterface
public String callNative(String name, String arguments) { public String callNative(String name, String arguments) {
JavaFunction javaFunction = globalFunctions.get(name); JavaFunction javaFunction = globalFunctions.get(name);
if (javaFunction == null) { if (javaFunction == null) {
return WRAPPED_NULL; DoricLog.e("Cannot find global function %s", name);
return WRAPPED_NULL_STRING;
} }
try { try {
JSONArray jsonArray = new JSONArray(arguments); JSONArray jsonArray = new JSONArray(arguments);
@ -86,33 +149,11 @@ public class DoricWebViewJSExecutor implements IDoricJSE {
decoders[i] = new JavaJSDecoder(object); decoders[i] = new JavaJSDecoder(object);
} }
JavaValue javaValue = javaFunction.exec(decoders); JavaValue javaValue = javaFunction.exec(decoders);
if (javaValue.getType() == 0) { return wrapJavaValue(javaValue).toString();
return WRAPPED_NULL;
}
if (javaValue.getType() == 1) {
Double value = Double.valueOf(javaValue.getValue());
return new JSONBuilder().put("type", "number").put("value", value).toString();
}
if (javaValue.getType() == 2) {
Boolean value = Boolean.valueOf(javaValue.getValue());
return new JSONBuilder().put("type", "boolean").put("value", value).toString();
}
if (javaValue.getType() == 3) {
String value = String.valueOf(javaValue.getValue());
return new JSONBuilder().put("type", "string").put("value", value).toString();
}
if (javaValue.getType() == 4) {
String value = String.valueOf(javaValue.getValue());
return new JSONBuilder().put("type", "object").put("value", value).toString();
}
if (javaValue.getType() == 5) {
String value = String.valueOf(javaValue.getValue());
return new JSONBuilder().put("type", "array").put("value", value).toString();
}
} catch (JSONException e) { } catch (JSONException e) {
e.printStackTrace(); e.printStackTrace();
} }
return WRAPPED_NULL; return WRAPPED_NULL_STRING;
} }
} }
@ -163,15 +204,40 @@ public class DoricWebViewJSExecutor implements IDoricJSE {
@Override @Override
public void injectGlobalJSFunction(String name, JavaFunction javaFunction) { public void injectGlobalJSFunction(String name, JavaFunction javaFunction) {
globalFunctions.put(name, javaFunction); globalFunctions.put(name, javaFunction);
loadJS(String.format("__injectGlobalFunction('%s')", name), "");
} }
@Override @Override
public void injectGlobalJSObject(String name, JavaValue javaValue) { public void injectGlobalJSObject(String name, JavaValue javaValue) {
loadJS(String.format("_injectGlobalObject('%s','%s')", name, javaValue.getValue()), "");
} }
@Override @Override
public JSDecoder invokeMethod(String objectName, String functionName, JavaValue[] javaValues, boolean hashKey) throws JSRuntimeException { public JSDecoder invokeMethod(String objectName, String functionName, JavaValue[] javaValues, boolean hashKey) throws JSRuntimeException {
JSONArray jsonArray = new JSONArray();
for (int i = 0; i < javaValues.length; i++) {
JavaValue javaValue = javaValues[i];
JSONObject jsonObject = wrapJavaValue(javaValue);
jsonArray.put(jsonObject);
}
returnFuture = new SettableFuture<>();
String script = String.format(
"__invokeMethod('%s','%s','%s')",
objectName,
functionName,
jsonArray.toString());
loadJS(script, "");
String result = returnFuture.get();
if (!TextUtils.isEmpty(result)) {
try {
JSONObject jsonObject = new JSONObject(result);
Object object = unwrapJSObject(jsonObject);
return new JavaJSDecoder(object);
} catch (JSONException e) {
e.printStackTrace();
}
}
return null; return null;
} }

View File

@ -1,2 +1,128 @@
'use strict'; 'use strict';
"use strict";
function _binaryValue(v) {
switch (typeof v) {
case "number":
return {
type: "number",
value: v
};
case "string":
return {
type: "string",
value: v
};
case "boolean":
return {
type: "boolean",
value: v
};
case "object":
if (v instanceof Array) {
return {
type: "array",
value: JSON.stringify(v)
};
}
else {
return {
type: "object",
value: JSON.stringify(v)
};
}
default:
return {
type: "null",
value: undefined
};
}
}
function _wrappedValue(v) {
switch (typeof v) {
case "number":
return {
type: "number",
value: v
};
case "string":
return {
type: "string",
value: v
};
case "boolean":
return {
type: "boolean",
value: v
};
case "object":
if (v instanceof Array) {
return {
type: "array",
value: JSON.stringify(v)
};
}
else {
return {
type: "object",
value: JSON.stringify(v)
};
}
default:
return {
type: "null",
value: undefined
};
}
}
function _rawValue(v) {
switch (v.type) {
case "number":
return v.value;
case "string":
return v.value;
case "boolean":
return v.value;
case "object":
case "array":
return v.value;
default:
return undefined;
}
}
function _injectGlobalObject(name, args) {
Reflect.set(window, name, JSON.parse(args));
}
function __injectGlobalFunction(name) {
Reflect.set(window, name, function () {
const args = [];
for (let i = 0; i < arguments.length; i++) {
args.push(_wrappedValue(arguments[i]));
}
const ret = NativeClient.callNative(name, JSON.stringify(args));
return _rawValue(JSON.parse(ret));
});
}
function __invokeMethod(objectName, functionName, stringifiedArgs) {
NativeClient.log(`invoke:${objectName}.${functionName}(${stringifiedArgs})`);
try {
const thisObject = Reflect.get(window, objectName);
const thisFunction = Reflect.get(thisObject, functionName);
const args = JSON.parse(stringifiedArgs);
args.forEach(e => {
NativeClient.log(`Arg:${e},${typeof e}`);
});
const rawArgs = args.map(e => _rawValue(e));
rawArgs.forEach(e => {
NativeClient.log(`RawArg:${e},${typeof e}`);
});
const ret = Reflect.apply(thisFunction, thisObject, rawArgs);
const returnVal = JSON.stringify(_wrappedValue(ret));
NativeClient.log(`return:${returnVal}`);
NativeClient.returnNative(returnVal);
}
catch (e) {
NativeClient.log(`error:${e},${e.stack}`);
NativeClient.returnNative("");
}
}

View File

@ -1,4 +1,6 @@
declare module NativeClient { declare module NativeClient {
function log(message: string): void
function returnNative(ret: string): void
function callNative(name: string, args: string): string function callNative(name: string, args: string): string
} }
@ -8,7 +10,43 @@ type WrappedValue = {
type: "number" | "string" | "boolean" | "object" | "array" | "null", type: "number" | "string" | "boolean" | "object" | "array" | "null",
value: RawValue, value: RawValue,
} }
function _binaryValue(v: RawValue) {
switch (typeof v) {
case "number":
return {
type: "number",
value: v
};
case "string":
return {
type: "string",
value: v
};
case "boolean":
return {
type: "boolean",
value: v
};
case "object":
if (v instanceof Array) {
return {
type: "array",
value: JSON.stringify(v)
};
} else {
return {
type: "object",
value: JSON.stringify(v)
};
}
default:
return {
type: "null",
value: undefined
};
}
}
function _wrappedValue(v: RawValue): WrappedValue { function _wrappedValue(v: RawValue): WrappedValue {
switch (typeof v) { switch (typeof v) {
case "number": case "number":
@ -49,20 +87,23 @@ function _wrappedValue(v: RawValue): WrappedValue {
function _rawValue(v: WrappedValue): RawValue { function _rawValue(v: WrappedValue): RawValue {
switch (v.type) { switch (v.type) {
case "number": case "number":
return v.value return v.value;
case "string": case "string":
return v.value return v.value;
case "boolean": case "boolean":
return v.value return v.value;
case "object": case "object":
case "array": case "array":
return JSON.stringify(v.value) if (typeof v.value === 'string') {
return JSON.parse(v.value)
}
return v.value;
default: default:
return undefined return undefined;
} }
} }
function __injectGlobalObject(name: string, args: string) { function _injectGlobalObject(name: string, args: string) {
Reflect.set(window, name, JSON.parse(args)); Reflect.set(window, name, JSON.parse(args));
} }
@ -73,6 +114,23 @@ function __injectGlobalFunction(name: string) {
args.push(_wrappedValue(arguments[i])); args.push(_wrappedValue(arguments[i]));
} }
const ret = NativeClient.callNative(name, JSON.stringify(args)); const ret = NativeClient.callNative(name, JSON.stringify(args));
return _rawValue(JSON.parse(ret)) return _rawValue(JSON.parse(ret));
}); });
} }
function __invokeMethod(objectName: string, functionName: string, stringifiedArgs: string) {
NativeClient.log(`invoke:${objectName}.${functionName}(${stringifiedArgs})`)
try {
const thisObject = Reflect.get(window, objectName);
const thisFunction = Reflect.get(thisObject, functionName);
const args = JSON.parse(stringifiedArgs) as WrappedValue[];
const rawArgs = args.map(e => _rawValue(e));
const ret = Reflect.apply(thisFunction, thisObject, rawArgs);
const returnVal = JSON.stringify(_wrappedValue(ret))
NativeClient.log(`return:${returnVal}`)
NativeClient.returnNative(returnVal)
} catch (e) {
NativeClient.log(`error:${e},${(e as any).stack}`)
NativeClient.returnNative("")
}
}

View File

@ -1,4 +1,6 @@
declare module NativeClient { declare module NativeClient {
function log(message: string): void;
function returnNative(ret: string): void;
function callNative(name: string, args: string): string; function callNative(name: string, args: string): string;
} }
declare type RawValue = number | string | boolean | object | undefined; declare type RawValue = number | string | boolean | object | undefined;
@ -6,7 +8,21 @@ declare type WrappedValue = {
type: "number" | "string" | "boolean" | "object" | "array" | "null"; type: "number" | "string" | "boolean" | "object" | "array" | "null";
value: RawValue; value: RawValue;
}; };
declare function _binaryValue(v: RawValue): {
type: string;
value: number;
} | {
type: string;
value: string;
} | {
type: string;
value: boolean;
} | {
type: string;
value: undefined;
};
declare function _wrappedValue(v: RawValue): WrappedValue; declare function _wrappedValue(v: RawValue): WrappedValue;
declare function _rawValue(v: WrappedValue): RawValue; declare function _rawValue(v: WrappedValue): RawValue;
declare function __injectGlobalObject(name: string, args: string): void; declare function _injectGlobalObject(name: string, args: string): void;
declare function __injectGlobalFunction(name: string): void; declare function __injectGlobalFunction(name: string): void;
declare function __invokeMethod(objectName: string, functionName: string, stringifiedArgs: string): void;

View File

@ -1,4 +1,41 @@
"use strict"; "use strict";
function _binaryValue(v) {
switch (typeof v) {
case "number":
return {
type: "number",
value: v
};
case "string":
return {
type: "string",
value: v
};
case "boolean":
return {
type: "boolean",
value: v
};
case "object":
if (v instanceof Array) {
return {
type: "array",
value: JSON.stringify(v)
};
}
else {
return {
type: "object",
value: JSON.stringify(v)
};
}
default:
return {
type: "null",
value: undefined
};
}
}
function _wrappedValue(v) { function _wrappedValue(v) {
switch (typeof v) { switch (typeof v) {
case "number": case "number":
@ -46,12 +83,12 @@ function _rawValue(v) {
return v.value; return v.value;
case "object": case "object":
case "array": case "array":
return JSON.stringify(v.value); return v.value;
default: default:
return undefined; return undefined;
} }
} }
function __injectGlobalObject(name, args) { function _injectGlobalObject(name, args) {
Reflect.set(window, name, JSON.parse(args)); Reflect.set(window, name, JSON.parse(args));
} }
function __injectGlobalFunction(name) { function __injectGlobalFunction(name) {
@ -64,3 +101,26 @@ function __injectGlobalFunction(name) {
return _rawValue(JSON.parse(ret)); return _rawValue(JSON.parse(ret));
}); });
} }
function __invokeMethod(objectName, functionName, stringifiedArgs) {
NativeClient.log(`invoke:${objectName}.${functionName}(${stringifiedArgs})`);
try {
const thisObject = Reflect.get(window, objectName);
const thisFunction = Reflect.get(thisObject, functionName);
const args = JSON.parse(stringifiedArgs);
args.forEach(e => {
NativeClient.log(`Arg:${e},${typeof e}`);
});
const rawArgs = args.map(e => _rawValue(e));
rawArgs.forEach(e => {
NativeClient.log(`RawArg:${e},${typeof e}`);
});
const ret = Reflect.apply(thisFunction, thisObject, rawArgs);
const returnVal = JSON.stringify(_wrappedValue(ret));
NativeClient.log(`return:${returnVal}`);
NativeClient.returnNative(returnVal);
}
catch (e) {
NativeClient.log(`error:${e},${e.stack}`);
NativeClient.returnNative("");
}
}

View File

@ -61,6 +61,7 @@ export default [
} }
console.warn(warning.message); console.warn(warning.message);
}, },
treeshake: false,
}, },
{ {
input: "lib-es5/index.runtime.es5.js", input: "lib-es5/index.runtime.es5.js",