我正在开发一个不可见的Java Applet,它将完全由JavaScript控制。
我可以轻松调用applet的Java方法,并且可以使用netscape.javascript.JSObject.getWindow(this).call()
从applet中调用JavaScript方法。
但是为了在applet中注册一个JavaScript回调,我想我需要某种JavaScript函数对象。
我想这样做:
public void registerCallback( SomeJavascriptFunction func ) { ... }
我可以通过Javascript调用:
myapplet.registerCallback(function(){ alert("called back"); });
所以我可以在以后的代码中调用这个函数:
func.call( ... );
这样的事情存在吗?我怎么能这样做?
Rigth现在我正在考虑创建一些Javascript来处理这个回调机制,而不是从applet中这样做。
答案 0 :(得分:4)
我意识到这是一个真的旧问题,但它在我的其他搜索中排名第二,我认为以下内容可能会帮助其他人找到这个。
我最近做了类似的事情,Java applet需要在完成任务时回调JavaScript,在成功或错误时调用不同的函数。正如最近的趋势一样,我的需求是调用定义为传递给其他函数的参数的匿名函数。这是客户端的javascript:
applet.DoProcessing({
success: function(param) {
alert('Success: ' + param);
},
error: function(param) {
alert('Failed: ' + param);
}
});
如其他答案所述,Java只能按名称调用JavaScript方法。这意味着您需要一个全局回调方法,然后可以根据需要调用其他方法:
function ProcessingCallback(isSuccessful, cbParam, jsObject) {
if (isSuccessful && jsObject.success)
jsObject.success(cbParam);
else if (!isSuccessful && jsObject.error)
jsObject.error(cbParam);
}
直接从Java applet中调用此函数:
public void DoProcessing(final Object callbacks) {
//do processing....
JSObject w = JSObject.getWindow(this);
//Call our named callback, note how we pass the callbacks parameter straight
//back out again - it will be unchanged in javascript.
w.call("ProcessingCallback", new Object[]{successful, output, callbacks});
}
如果你想将它用作某种形式的注册回调而不是一次性的回调函数,你可以保留无限期传递的参数对象的引用。
在我们的情况下,处理可能是时间紧张的,所以我们实际上启动另一个线程 - 回调仍然在这里工作:
public void DoProcessing(final Object callbacks) {
//hold a reference for use in the thread
final Applet app = this;
//Create a new Thread object to run our request asynchronously
//so we can return back to single threaded javascript immediately
Thread async = new Thread() {
//Thread objects need a run method
public void run() {
//do processing....
JSObject w = JSObject.getWindow(app);
//Call our named callback, note how we pass the callbacks parameter
//straight back out again - it will be unchanged in javascript.
w.call("ProcessingCallback", new Object[]{successful, output, callbacks});
}
}
//start the thread
async.start();
}
答案 1 :(得分:2)
我是Java的新手< - >正如我计划在本周探索它一样进行JavaScript通信。这是个好机会......: - )
经过一些测试,似乎你无法将JS函数传递给Java applet。除非我以错误的方式做到这一点......
我试过了:
function CallJava()
{
document.Applet.Call("Does it work?");
document.Applet.Call(function () { alert("It works!"); });
document.Applet.Call(DoSomething); // A simple parameterless JS function
document.Applet.Call(window.location);
}
function DumbTest(message, value)
{
alert("This is a dumb test with a message:\n" + message + "\n" + value);
}
其中Call被定义为:
public void Call(String message)
{
JSObject win = (JSObject) JSObject.getWindow(this);
String[] arguments = { "Call with String", message };
win.call("DumbTest", arguments);
}
public void Call(JSObject jso)
{
JSObject win = (JSObject) JSObject.getWindow(this);
String[] arguments = { "Call with JSObject", jso.toString() };
win.call("DumbTest", arguments);
}
当我传递一个JS函数(FF3中的所有测试)时,我在Java端得到一个null。
请注意,以下Java例程允许显示DumberTest函数的JS代码!
public int Do()
{
JSObject win = (JSObject) JSObject.getWindow(this);
JSObject doc = (JSObject) win.getMember("document");
JSObject fun = (JSObject) win.getMember("DumberTest");
JSObject loc = (JSObject) doc.getMember("location");
String href = (String) loc.getMember("href");
String[] arguments = { href, fun.toString() };
win.call("DumbTest", arguments);
return fun.toString().length();
}
重点:我做了一个JS函数:
function RegisterCallback(cbFunction)
{
var callback = cbFunction.toString(); // We get JS code
var callbackName = /^function (\w+)\(/.exec(callback);
document.Applet.RegisterCallback(callbackName[1]);
}
我从toString结果中提取JS函数的名称,并将其传递给Java applet。我认为我们不能处理匿名函数,因为Java按名称调用JS函数。
Java方面:
String callbackFunction;
public void RegisterCallback(String functionName)
{
callbackFunction = functionName;
}
void UseCallbackFunction()
{
if (callbackFunction == null) return;
JSObject win = (JSObject) JSObject.getWindow(this);
win.call(callbackFunction, null);
}
答案 2 :(得分:1)
win.eval()将调用预定义的javascript。
String callbackFunction;
public void RegisterCallback(String functionName)
{
callbackFunction = functionName;
}
void UseCallbackFunction()
{
if (callbackFunction == null) return;
JSObject win = (JSObject) JSObject.getWindow(this);
win.eval(callbackFunction);
}