All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.codename1.ui.BrowserComponent Maven / Gradle / Ivy

/*
 * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores
 * CA 94065 USA or visit www.oracle.com if you need additional information or
 * have any questions.
 */
package com.codename1.ui;

import com.codename1.io.JSONParser;
import com.codename1.io.Log;
import com.codename1.io.URL;
import com.codename1.io.Util;
import com.codename1.processing.Result;
import com.codename1.ui.events.ActionEvent;
import com.codename1.ui.events.ActionListener;
import com.codename1.ui.layouts.BorderLayout;
import com.codename1.ui.util.EventDispatcher;
import com.codename1.ui.events.BrowserNavigationCallback;
import com.codename1.ui.plaf.Style;
import com.codename1.ui.util.UITimer;
import com.codename1.util.AsyncResource;
import com.codename1.util.Base64;
import com.codename1.util.Callback;
import com.codename1.util.CallbackAdapter;
import com.codename1.util.StringUtil;
import com.codename1.util.SuccessCallback;
import com.codename1.util.regex.StringReader;
import java.io.IOException;
import java.net.URI;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;

import java.util.Hashtable;
import java.util.LinkedList;
import java.util.Map;
import java.util.Set;
import java.util.Timer;
import java.util.Vector;

/**
 * 

The browser component is an interface to an embeddable native platform browser on platforms * that support embedding the native browser in place, if you need wide compatibility and flexibility * you should check out the HTMLComponent which provides a lightweight 100% cross platform * web component.
* This component will only work on platforms that support embedding a native browser which * exclude earlier versions of Blackberry devices and J2ME devices.
* Its recommended that you place this component in a fixed position (none scrollable) on the screen without other * focusable components to prevent confusion between focus authority and allow the component to scroll * itself rather than CodenameOne making that decision for it.

* *

On Android this component might show a native progress indicator dialog. You can disable that functionality * using the {@Display.getInstance().setProperty("WebLoadingHidden", "true");} call.

* *

* The following code shows the basic usage of the {@code BrowserComponent}: *

* * Simple usage of BrowserComponent * *

Debugging on Android

* *

You can use Chrome's remote debugging features to debug the contents of a BrowserComponent. On Android 4.4 (KitKat) * and higher, you will need to define the "android.webContentsDebuggingEnabled" display property in order for this to work. You can define this inside your app's init() method:

*
Display.getInstance().setProperty("android.webContentsDebuggingEnabled", "true");
* @author Shai Almog */ public class BrowserComponent extends Container { private Hashtable listeners; private PeerComponent internal; private boolean pinchToZoom = true; private boolean nativeScrolling = true; private boolean ready = false; private boolean fireCallbacksOnEdt=true; PeerComponent getInternal() { return internal; } /** * String constant for web event listener {@link #addWebEventListener(java.lang.String, com.codename1.ui.events.ActionListener)} */ public static final String onStart = "onStart"; /** * String constant for web event listener {@link #addWebEventListener(java.lang.String, com.codename1.ui.events.ActionListener)} */ public static final String onLoad = "onLoad"; /** * String constant for web event listener {@link #addWebEventListener(java.lang.String, com.codename1.ui.events.ActionListener)} */ public static final String onError = "onError"; /** * String constant for web event listener. Use this event types to register to receive messages * in a cross-domain-safe way from the web page. To send a message from the webpage, the page should * include a function like: *
{@code 
     * function postToCN1(msg) {
     *       if (window.cn1PostMessage) {
     *           // Case 1: Running inside native app in a WebView
     *           window.cn1PostMessage(msg);
     *       } else {
     *           // Case 2: Running inside a Javascript app in an iframe
     *           window.parent.postMessage(msg, '*');
     *       }
     *   }
     * }
*

Receiving a message:

*
{@code 
     * myBrowserComponent.addWebEventListener(BrowserComponent.onMessage, e->{
     *       CN.callSerially(()->{
     *           Log.p("Message: "+e.getSource());
     *           Dialog.show("Here", (String)e.getSource(), "OK", null);
     *       });
     *   });
     * }
*/ public static final String onMessage = "onMessage"; private BrowserNavigationCallback browserNavigationCallback = new BrowserNavigationCallback(){ public boolean shouldNavigate(String url) { return true; } }; /** * Sets whether javascript callbacks should be run on the EDT. Default is {@literal true}. * @param edt True if callbacks should be run on EDT. False if they should be run on the platform's main thread. * @since 5.0 */ public void setFireCallbacksOnEdt(boolean edt) { this.fireCallbacksOnEdt = edt; } /** * Checks if javascript callbacks are run on the EDT. * @return True if javascript callbacks are run on the EDT. * @since 5.0 * @see #setFireCallbacksOnEdt(boolean) */ public boolean isFireCallbacksOnEdt() { return fireCallbacksOnEdt; } /** * Set the browser navigation callback which allows handling a case where * a URL invocation can be delegated to Java code. This allows binding * Java side functionality to JavaScript functionality in the same * way PhoneGap/Cordova work * @param callback the callback interface * @deprecated Use {@link #addBrowserNavigationCallback(com.codename1.ui.events.BrowserNavigationCallback) Instead */ public void setBrowserNavigationCallback(BrowserNavigationCallback callback){ this.browserNavigationCallback = callback; } /** * Async method for capturing a screenshot of the browser content. Currently only supported * in the simulator. Also, only displays the visible rectangle of the BrowserComponent, * not the entire page. * @return AsyncResource resolving to an Image of the webview contents. * @since 7.0 */ public AsyncResource captureScreenshot() { if (internal != null) { AsyncResource i = Display.impl.captureBrowserScreenshot(internal); if (i != null) { return i; } } AsyncResource out = new AsyncResource(); if (internal != null) { out.complete(internal.toImage()); } else { out.complete(toImage()); } return out; } /** * The browser navigation callback interface allows handling a case where * a URL invocation can be delegated to Java code. This allows binding * Java side functionality to JavaScript functionality in the same * way PhoneGap/Cordova work * * @return the callback interface * @deprecated Call {@link #fireBrowserNavigationCallbacks(java.lang.String) } to determine whether navigation should occur for a particulr URL. */ public BrowserNavigationCallback getBrowserNavigationCallback(){ return this.browserNavigationCallback; } /** * List of registered browser navigation callbacks. */ private Vector browserNavigationCallbacks; private Vector browserNavigationCallbacks() { if (browserNavigationCallbacks == null) { browserNavigationCallbacks = new Vector(); } return browserNavigationCallbacks; } /** * Adds a navigation callback. * @param callback The callback to call before navigating to a URL. */ public void addBrowserNavigationCallback(BrowserNavigationCallback callback) { browserNavigationCallbacks().add(callback); } /** * Removes a navigation callback. * @param callback The callback to call before navigating to a URL. */ public void removeBrowserNavigationCallback(BrowserNavigationCallback callback) { if (browserNavigationCallbacks != null) { browserNavigationCallbacks().remove(callback); } } private static final String RETURN_URL_PREFIX = "/!cn1return/"; private Hashtable> returnValueCallbacks; private Hashtable> returnValueCallbacks() { if (returnValueCallbacks == null) { returnValueCallbacks = new Hashtable>(); } return returnValueCallbacks; } private int nextReturnValueCallbackId = 0; private int addReturnValueCallback(SuccessCallback callback) { int id = nextReturnValueCallbackId++; while (returnValueCallbacks().containsKey(id)) { id++; } returnValueCallbacks().put(id, callback); nextReturnValueCallbackId = id+1; if (nextReturnValueCallbackId > 10000) { nextReturnValueCallbackId=0; } return id; } private SuccessCallback popReturnValueCallback(int id) { if (returnValueCallbacks != null) { return returnValueCallbacks.remove(id); } return null; } private JSONParser returnValueParser; private JSONParser returnValueParser() { if (returnValueParser == null) { returnValueParser = new JSONParser(); } return returnValueParser; } @Override protected void initComponent() { super.initComponent(); } @Override protected void deinitialize() { uninstallMessageListener(); super.deinitialize(); } /** * Calls the {@literal postMessage()} method on the webpage's {@literal window} object. *

This is useful mainly for the Javascript port so that you don't have to worry about * cross-domain issues, as postMessage() is supported cross-domain.

* *

To receive a message, the web page should register a "message" event listener, just as * it would to receive messages from other windows in the browser. See MDN docs for postMessage() * for more information.

* * @param message The message to send. * @param targetOrigin The target origin of the message. E.g. http://example.com:1234 * @since 7.0 */ public void postMessage(final String message, final String targetOrigin) { if (internal == null) { onReady(new Runnable() { public void run() { postMessage(message, targetOrigin); } }); return; } if (!Display.impl.postMessage(internal, message, targetOrigin)) { execute("window.postMessage(${0}, ${1})", new Object[]{message, targetOrigin}); } } private void installMessageListener() { if (internal == null) { onReady(new Runnable() { public void run() { installMessageListener(); } }); return; } if (!Display.impl.installMessageListener(internal)) { messageCallback = new SuccessCallback() { @Override public void onSucess(JSRef value) { fireWebEvent(onMessage, new ActionEvent(value.toString())); } }; addJSCallback("window.cn1PostMessage = function(msg){ callback.onSuccess(msg);};", messageCallback); } } SuccessCallback messageCallback; private void uninstallMessageListener() { if (internal == null) { onReady(new Runnable() { public void run() { uninstallMessageListener(); } }); return; } if (!Display.impl.installMessageListener(internal)) { //if (messageCallback != null) { // removeJSCallback(messageCallback); // messageCallback = null; //} } } /** * Decodes a URL * @param s The string to decode. * @param enc The encoding. E.g. UTF-8 * @return The decoded URL. */ private static String decodeURL(String s, String enc){ boolean needToChange = false; int numChars = s.length(); StringBuilder sb = new StringBuilder(numChars > 500 ? numChars / 2 : numChars); int i = 0; char c; byte[] bytes = null; while (i < numChars) { c = s.charAt(i); switch (c) { case '+': sb.append(' '); i++; needToChange = true; break; case '%': /* * Starting with this instance of %, process all * consecutive substrings of the form %xy. Each * substring %xy will yield a byte. Convert all * consecutive bytes obtained this way to whatever * character(s) they represent in the provided * encoding. */ try { // (numChars-i)/3 is an upper bound for the number // of remaining bytes if (bytes == null) bytes = new byte[(numChars-i)/3]; int pos = 0; while ( ((i+2) < numChars) && (c=='%')) { int v = Integer.parseInt(s.substring(i+1,i+3),16); if (v < 0) throw new IllegalArgumentException("URLDecoder: Illegal hex characters in escape (%) pattern - negative value"); bytes[pos++] = (byte) v; i+= 3; if (i < numChars) c = s.charAt(i); } // A trailing, incomplete byte encoding such as // "%x" will cause an exception to be thrown if ((i < numChars) && (c=='%')) throw new IllegalArgumentException( "URLDecoder: Incomplete trailing escape (%) pattern"); try { sb.append(new String(bytes, 0, pos, enc)); } catch (Throwable t) { throw new RuntimeException(t.getMessage()); } } catch (NumberFormatException e) { throw new IllegalArgumentException( "URLDecoder: Illegal hex characters in escape (%) pattern - " + e.getMessage()); } needToChange = true; break; default: sb.append(c); i++; break; } } return (needToChange? sb.toString() : s); } /** * Fires all of the registered browser navigation callbacks against the provided URL. * @param url The URL to fire the navigation callbacks against. * @return True if all of the callbacks say that they can browse. False otherwise. */ public boolean fireBrowserNavigationCallbacks(String url) { boolean shouldNavigate = true; if (browserNavigationCallback != null && !browserNavigationCallback.shouldNavigate(url)) { shouldNavigate = false; } if (browserNavigationCallbacks != null) { for (BrowserNavigationCallback cb : browserNavigationCallbacks) { if (!cb.shouldNavigate(url)) { shouldNavigate = false; } } } if ( !url.startsWith("javascript:") && url.indexOf(RETURN_URL_PREFIX) != -1 ){ //System.out.println("Received browser navigation callback "+url); String result = decodeURL(url.substring(url.indexOf(RETURN_URL_PREFIX) + RETURN_URL_PREFIX.length()), "UTF-8"); //System.out.println("After decode "+result); Result structResult = Result.fromContent(result, Result.JSON); int callbackId = structResult.getAsInteger("callbackId"); final String value = structResult.getAsString("value"); final String type = structResult.getAsString("type"); final String errorMessage = structResult.getAsString("errorMessage"); final SuccessCallback callback = popReturnValueCallback(callbackId); if (jsCallbacks != null && jsCallbacks.contains(callback)) { // If this is a registered callback, then we treat it more like // an event listener, and we retain it for future callbacks. returnValueCallbacks.put(callbackId, callback); } if (callback != null) { if (errorMessage != null) { if (fireCallbacksOnEdt) { Display.getInstance().callSerially(new Runnable() { public void run() { if (callback instanceof Callback) { ((Callback)callback).onError(this, new RuntimeException(errorMessage), 0, errorMessage); } } }); } else { if (callback instanceof Callback) { ((Callback)callback).onError(this, new RuntimeException(errorMessage), 0, errorMessage); } } } else { if (fireCallbacksOnEdt) { Display.getInstance().callSerially(new Runnable() { public void run() { callback.onSucess(new JSRef(value, type)); } }); } else { callback.onSucess(new JSRef(value, type)); } } } else { Log.e(new RuntimeException("Received return value from javascript, but no callback could be found for that ID")); } shouldNavigate = false; } return shouldNavigate; } private Container placeholder = new Container(); private LinkedList onReady = new LinkedList(); private void onReady(final Runnable r) { if (!CN.isEdt()) { CN.callSerially(new Runnable() { public void run() { onReady(r); } }); return; } onReady.add(r); if (internal != null) { while (!onReady.isEmpty()) { onReady.remove(0).run(); } } } private void onReady() { if (internal != null) { while (!onReady.isEmpty()) { onReady.remove(0).run(); } } } /** * This constructor will work as expected when a browser component is supported, see isNativeBrowserSupported() */ public BrowserComponent() { setUIID("BrowserComponent"); putClientProperty("BrowserComponent.useWKWebView", "true".equals(Display.getInstance().getProperty("BrowserComponent.useWKWebView", "true"))); setLayout(new BorderLayout()); addComponent(BorderLayout.CENTER, placeholder); CN.callSerially(new Runnable() { public void run() { PeerComponent c = Display.impl.createBrowserComponent(BrowserComponent.this); if (c == null) { if (CN.isSimulator()) { Log.p("Failed to create the browser component. Please ensure that you are either using a JDK that has JavaFX (e.g. ZuluFX), or that you have installed the Codename One CEF component. See https://www.codenameone.com/blog/big-changes-jcef.html for more information"); } else { Log.p("Failed to create browser component. This platform may not support the native browser component"); } return; } internal = c; removeComponent(placeholder); addComponent(BorderLayout.CENTER, internal); onReady(); revalidateLater(); } }); onReady(new Runnable() { public void run() { Style s = internal.getUnselectedStyle(); s.setPadding(0, 0, 0, 0); s.setMargin(0, 0, 0, 0); s.setBgTransparency(255); s = getUnselectedStyle(); s.setPadding(0, 0, 0, 0); } }); addWebEventListener(onStart, new ActionListener() { @Override public void actionPerformed(ActionEvent evt) { installMessageListener(); } }); } private final Object readyLock = new Object(); /** * Uses invokeAndBlock to wait until the BrowserComponent is ready. The browser component * is considered to be ready once the onLoad event has been fired for the first page. */ public void waitForReady() { while (!ready) { Display.getInstance().invokeAndBlock(new Runnable() { public void run() { synchronized(readyLock) { Util.wait(readyLock, 1000); } } }); } } /** * Registers a callback to be run when the BrowserComponent is "ready". The browser component * is considered to be ready once the onLoad event has been fired on the first page. * If this method is called after the browser component is already "ready", then the callback * will be executed immediately. Otherwise it will be called in the first onLoad event. * @param onReady Callback to be executed when the browser component is ready. * @return Self for chaining. * @since 7.0 * @see #waitForReady() */ public BrowserComponent ready(final SuccessCallback onReady) { if (ready) { onReady.onSucess(this); } else { ActionListener l = new ActionListener() { public void actionPerformed(ActionEvent evt) { removeWebEventListener(onStart, this); onReady.onSucess(BrowserComponent.this); } }; addWebEventListener(onStart, l); } return this; } /** * Returns a promise that will complete when the browser component is "ready". It is considered to be * ready once it has received the start or load event from at least one page. Default timeout is 5000ms. * @return AsyncResouce that will complete when the browser component is ready. * @since 7.0 */ public AsyncResource ready() { return ready(5000); } /** * Returns a promise that will complete when the browser component is "ready". It is considered to be * ready once it has received the start or load event from at least one page. * @return AsyncResouce that will complete when the browser component is ready. * @param timeout Timeout in milliseconds to wait. * @since 7.0 */ public AsyncResource ready(int timeout) { final AsyncResource out = new AsyncResource(); if (ready) { out.complete(this); } else { class LoadWrapper { Timer timer; ActionListener l; } final LoadWrapper w = new LoadWrapper(); if (timeout > 0) { w.timer = CN.setTimeout(timeout, new Runnable(){ public void run() { w.timer = null; if (w.l != null) { removeWebEventListener(onStart, w.l); removeWebEventListener(onLoad, w.l); } if (!out.isDone()) { out.error(new RuntimeException("Timeout exceeded waiting for browser component to be ready")); } } }); } w.l = new ActionListener() { public void actionPerformed(ActionEvent evt) { w.l = null; if (w.timer != null) { w.timer.cancel(); w.timer = null; } removeWebEventListener(onStart, this); removeWebEventListener(onLoad, this); if (!out.isDone()) { out.complete(BrowserComponent.this); } } }; addWebEventListener(onStart, w.l); addWebEventListener(onLoad, w.l); } return out; } /** * Returns true if the platform supports embedding a native browser component * * @return true if native browsing is supported */ public static boolean isNativeBrowserSupported() { return Display.impl.isNativeBrowserComponentSupported(); } /** * This method allows customizing the properties of a web view in various ways including platform specific settings. * When a property isn't supported by a specific platform it is just ignored. * * @param key see the documentation with the CodenameOne Implementation for further details * @param value see the documentation with the CodenameOne Implementation for further details */ public void setProperty(final String key, final Object value) { if (internal == null) { onReady(new Runnable() { public void run() { setProperty(key, value); } }); return; } Display.impl.setBrowserProperty(internal, key, value); } /** * The page title * @return the title */ public String getTitle() { if (internal == null) { return null; } return Display.impl.getBrowserTitle(internal); } /** * The page URL * @return the URL */ public String getURL() { if (internal == null) { return tmpUrl; } return Display.impl.getBrowserURL(internal); } private String tmpUrl; /** * Sets the page URL, jar: URL's must be supported by the implementation * @param url the URL */ public void setURL(final String url) { if (internal == null) { tmpUrl = url; onReady(new Runnable() { public void run() { setURL(url); } }); return; } Display.impl.setBrowserURL(internal, url); } /** * Sets the page URL. * @param url The URL to the page to display. */ public void setURL(URL url) { setURL(url.toString()); } /** * Sets the page URL. * @param uri URI to the page to display. */ public void setURL(URI uri) { setURL(uri.toString()); } /** * Sets the page URL, jar: URL's must be supported by the implementation. Notice this API isn't supported * in all platforms see {@link #isURLWithCustomHeadersSupported() } * @param url the URL * @param headers headers to push into the request for the url */ public void setURL(final String url, final Map headers) { if (internal == null) { tmpUrl = url; onReady(new Runnable() { public void run() { setURL(url, headers); } }); return; } Display.impl.setBrowserURL(internal, url, headers); } /** * Returns true if the method {@link #setURL(java.lang.String, java.util.Map) } is supported * @return false by default */ public boolean isURLWithCustomHeadersSupported() { return Display.impl.isURLWithCustomHeadersSupported(); } /** * Sets the page URL while respecting the hierarchy of the html * @param url the URL */ public void setURLHierarchy(final String url) throws IOException { if (internal == null) { onReady(new Runnable() { public void run() { try { setURLHierarchy(url); } catch (IOException ex) { Log.e(ex); } } }); return; } Display.impl.setBrowserPageInHierarchy(internal, url); } /** * Reload the current page */ public void reload() { if (internal == null) { onReady(new Runnable() { public void run() { reload(); } }); return; } Display.impl.browserReload(internal); } /** * Indicates whether back is currently available * @return true if back should work */ public boolean hasBack() { if (internal == null) { return false; } return Display.impl.browserHasBack(internal); } /** * Indicates whether forward is currently available * @return true if forward should work */ public boolean hasForward() { if (internal == null) { return false; } return Display.impl.browserHasForward(internal); } /** * Navigates back in the history */ public void back() { if (internal == null) { onReady(new Runnable() { public void run() { back(); } }); return; } Display.impl.browserBack(internal); } /** * Navigates forward in the history */ public void forward() { if (internal == null) { onReady(new Runnable() { public void run() { forward(); } }); return; } Display.impl.browserForward(internal); } /** * Clears navigation history */ public void clearHistory() { if (internal == null) { onReady(new Runnable() { public void run() { clearHistory(); } }); return; } Display.impl.browserClearHistory(internal); } /** * Some platforms require that you enable pinch to zoom explicitly. This method has no * effect if pinch to zoom isn't supported by the platform * * @param e true to enable pinch to zoom, false to disable it */ public void setPinchToZoomEnabled(final boolean e) { pinchToZoom = e; if (internal == null) { onReady(new Runnable() { public void run() { setPinchToZoomEnabled(e); } }); return; } Display.impl.setPinchToZoomEnabled(internal, e); } /** * This method is unreliable and is only here for consistency with setPinchToZoomEnabled, * it will not return whether the platform supports pinch since this is very hard to detect * properly. * @return the last value for setPinchToZoomEnabled */ public boolean isPinchToZoomEnabled() { return pinchToZoom; } /** * This flag allows disabling the native browser scrolling on platforms that support it * @param b true to enable native scrolling, notice that non-native scrolling might be problematic */ public void setNativeScrollingEnabled(final boolean b) { nativeScrolling = b; if (internal == null) { onReady(new Runnable() { public void run() { setNativeScrollingEnabled(b); } }); return; } Display.impl.setNativeBrowserScrollingEnabled(internal, b); } /** * This method is unreliable and is only here for consistency with setNativeScrollingEnabled. * * @return the last value for setNativeScrollingEnabled */ public boolean isNativeScrollingEnabled() { return nativeScrolling; } /** * Shows the given HTML in the native viewer * * @param html HTML web page * @param baseUrl base URL to associate with the HTML */ public void setPage(final String html, final String baseUrl) { if (internal == null) { onReady(new Runnable() { public void run() { setPage(html, baseUrl); } }); return; } Display.impl.setBrowserPage(internal, html, baseUrl); } private EventDispatcher getEventDispatcher(String type, boolean autoCreate) { if(listeners == null) { if(!autoCreate) { return null; } listeners = new Hashtable(); EventDispatcher ev = new EventDispatcher(); listeners.put(type, ev); return ev; } EventDispatcher ev = (EventDispatcher)listeners.get(type); if(ev == null) { if(autoCreate) { ev = new EventDispatcher(); listeners.put(type, ev); } } return ev; } /** * Adds a listener to the given event type name, event type names are platform specific but some * must be fired for all platforms and will invoke the action listener when the appropriate event loads * * @param type platform specific but must support: onStart, onLoad, onError * @param listener callback for the event */ public void addWebEventListener(String type, ActionListener listener) { getEventDispatcher(type, true).addListener(listener); } /** * Removes the listener, see addWebEventListener for details * * @param type see addWebEventListener for details * @param listener see addWebEventListener for details */ public void removeWebEventListener(String type, ActionListener listener) { EventDispatcher e = getEventDispatcher(type, false); if(e != null) { e.removeListener(listener); if(!e.hasListeners()) { listeners.remove(type); } } } /** * Cancel the loading of the current page */ public void stop() { if (internal == null) { onReady(new Runnable() { public void run() { stop(); } }); return; } Display.impl.browserStop(internal); } /** * Release native resources of this Browser Component */ public void destroy() { if (internal == null) { onReady(new Runnable() { public void run() { destroy(); } }); return; } Display.impl.browserDestroy(internal); } /** * Used internally by the implementation to fire an event from the native browser widget * * @param type the type of the event * @param ev the event */ public void fireWebEvent(String type, ActionEvent ev) { if (onLoad.equals(type)) { synchronized(readyLock) { ready = true; readyLock.notifyAll(); } } EventDispatcher e = getEventDispatcher(type, false); if(e != null) { e.fireActionEvent(ev); } } /** * Executes the given JavaScript string within the current context * * @param javaScript the JavaScript string */ public void execute(final String javaScript) { if (internal == null) { onReady(new Runnable() { public void run() { execute(javaScript); } }); return; } Display.impl.browserExecute(internal, javaScript); } /** * Executes given javascript string within current context. * @param js The javascript to execute. * @param params Parameters to inject into the javascript expression. The expression should contain placeholders of the form {@literal ${0} }, {@literal ${1} }, etc... to be replaced. See {@link #injectParameters(java.lang.String, java.lang.Object...) } for more information about injected parameters. * by parameters. */ public void execute(String js, Object[] params) { execute(injectParameters(js, params)); } /** * Executes the given JavaScript and returns a result string from the underlying platform * where applicable. *

Note: Some platforms use {@link Display#invokeAndBlock(java.lang.Runnable) } inside this method which is very costly. Try to avoid this synchronous method, and * prefer to use one of the asynchronous versions. E.g. {@link #execute(java.lang.String, com.codename1.util.SuccessCallback) }

* @param javaScript the JavaScript code to execute * @return the string returned from the Javascript call */ public String executeAndReturnString(String javaScript){ if (internal == null) { while (internal == null) { CN.invokeAndBlock(new Runnable() { public void run() { Util.sleep(50); } }); } } if (Display.impl.supportsBrowserExecuteAndReturnString(internal)) { return Display.impl.browserExecuteAndReturnString(internal, javaScript); } else { return executeAndWait("callback.onSuccess(eval(${0}))", new Object[]{javaScript}).toString(); } } /** * Executes the given javascript and returns the result string from the underlying platform. *

Note: Some platforms use {@link Display#invokeAndBlock(java.lang.Runnable) } inside this method which is very costly. Try to avoid this synchronous method, and * prefer to use one of the asynchronous versions. E.g. {@link #execute(java.lang.String, com.codename1.util.SuccessCallback) }

* @param javaScript The javascript to execute. * @param params Parameters to inject into the javascript expression. The expression should contain placeholders of the form {@literal ${0} }, {@literal ${1} }, etc... to be replaced. See {@link #injectParameters(java.lang.String, java.lang.Object...) } for more information about injected parameters. * @return The result as a string. * @since 5.0 */ public String executeAndReturnString(String javaScript, Object[] params) { return executeAndReturnString(injectParameters(javaScript, params)); } /** * Creates a proxy for a Javascript object that makes it easier to call methods, retrieve, * and manipulate properties on the object. */ public JSProxy createJSProxy(String javascriptExpression) { return new JSProxy(javascriptExpression); } /** * A thin wrapper around a Javascript variable that makes it easier to * call methods on that variable. */ public class JSProxy { /** The javascript variable name. This can be any javascript expression that resolves * to an object. */ private final String self; /** * Creats a new proxy. * @param self The javascript expression that should resolve to the object that this * will proxy. E.g. "window", or "document.getElementById('mybutton')". The expression * is just stored as a string and is 'resolved' when calls are made on the proxy. */ private JSProxy(String self) { this.self = self; } /** * Calls a method on this javascript object. * @param timeout The timeout in ms * @param method The method name. * @param args Arguments to pass to the method. * @param callback Callback with the result of the method. */ public void call(int timeout, String method, Object[] args, SuccessCallback callback) { StringBuilder js = new StringBuilder(); js.append("callback.onSuccess("+self+"."+method+"("); int len = args.length; for (int i=0; i 0) { js.append(", "); } js.append("${"+i+"}"); } js.append("))"); execute(js.toString(), args, callback); } /** * Calls a method on this javascript object. * @param method The method name. * @param args Arguments to pass to the method. * @param callback Callback with the result of the method. */ public void call(String method, Object[] args, SuccessCallback callback) { call(0, method, args, callback); } /** * Calls method on this javascript object and waits for the result using * invokeAndBlock. * @param timeout The timeout in ms * @param method The method name. * @param args Arguments for the method. * @return JSRef with the result of the method call. */ public JSRef callAndWait(int timeout, String method, Object[] args) { StringBuilder js = new StringBuilder(); js.append("callback.onSuccess("+self+"."+method+"("); int len = args.length; for (int i=0; i 0) { js.append(", "); } js.append("${"+i+"}"); } js.append("))"); return executeAndWait(timeout, js.toString(), args); } /** * Calls method on this javascript object and waits for the result using * invokeAndBlock. * @param method The method name. * @param args Arguments for the method. * @return JSRef with the result of the method call. */ public JSRef callAndWait(String method, Object[] args) { return callAndWait(0, method, args); } /** * Gets a property of this javascript object. * @param timeout Timeout in ms * @param property The property name. * @param callback Callback with the property value. */ public void get(int timeout, String property, SuccessCallback callback) { StringBuilder js = new StringBuilder(); js.append("callback.onSuccess("+self+"."+property+")"); execute(timeout, js.toString(), callback); } /** * Gets a property of this javascript object. * @param property The property name. * @param callback Callback with the property value. */ public void get(String property, SuccessCallback callback) { get(0, property, callback); } /** * Gets multiple properties as a batch. * @param timeout Timeout in ms * @param properties List of property names to retrieve. * @param callback */ public void get(int timeout, Collection properties, final SuccessCallback> callback) { StringBuilder sb = new StringBuilder(); sb.append("(function(){var outmap={};var prop=null; var propval=null;"); for (String prop : properties) { sb.append("prop='").append(prop).append("';"); sb.append("propval=").append(self).append("[prop]"); sb.append("outmap[prop] = {value:propval, type:typeof(propval)};"); } sb.append("callback.onSuccess(JSON.stringify(outmap))})()"); execute(timeout, sb.toString(), new SuccessCallback() { public void onSucess(JSRef value) { JSONParser p = new JSONParser(); try { Map m = p.parseJSON(new StringReader(value.getValue())); Map out = new HashMap(); for (String prop : out.keySet()) { Map propVal = (Map)m.get(prop); out.put(prop, new JSRef((String)propVal.get("value"), (String)propVal.get("type"))); } callback.onSucess(out); } catch (Exception ex) { Log.e(ex); if (callback instanceof Callback) { ((Callback)callback).onError(BrowserComponent.this, ex, 0, ex.getMessage()); } else { callback.onSucess(null); } } } }); } /** * Gets a property of this javascript object and waits for the result * using invokeAndBlock. * @param property The property to retrieve. * @return The property value. */ public void get(Collection properties, final SuccessCallback> callback) { get(0, properties, callback); } /** * Gets a property of this javascript object and waits for the result * using invokeAndBlock. * @param timeout The timeout in ms * @param property The property to retrieve. * @return The property value. */ public JSRef getAndWait(int timeout, String property) { StringBuilder js = new StringBuilder(); js.append("callback.onSuccess("+self+"."+property+")"); return executeAndWait(timeout, js.toString()); } /** * Gets a property of this javascript object and waits for the result * using invokeAndBlock. * @param property The property to retrieve. * @return The property value. */ public JSRef getAndWait(String property) { return getAndWait(0, property); } /** * Gets multiple properties on this object in a batch. * @param timeout The timeout in ms * @param properties The property names to get. * @return */ public Map getAndWait(int timeout, Collection properties) { StringBuilder sb = new StringBuilder(); sb.append("(function(){var outmap={};var prop=null; var propval=null;"); for (String prop : properties) { sb.append("prop='").append(prop).append("';"); sb.append("propval=").append(self).append("[prop]"); sb.append("outmap[prop] = {value:propval, type:typeof(propval)};"); } sb.append("callback.onSuccess(JSON.stringify(outmap))})()"); JSRef value = executeAndWait(timeout, sb.toString()); JSONParser p = new JSONParser(); try { Map m = p.parseJSON(new StringReader(value.getValue())); Map out = new HashMap(); for (String prop : out.keySet()) { Map propVal = (Map)m.get(prop); out.put(prop, new JSRef((String)propVal.get("value"), (String)propVal.get("type"))); } return out; } catch (Exception ex) { if (ex instanceof RuntimeException) { throw (RuntimeException)ex; } else { Log.e(ex); throw new RuntimeException(ex.getMessage()); } } } /** * Gets multiple properties on this object in a batch. * @param properties * @return */ public Map getAndWait(Collection properties) { return getAndWait(0, properties); } /** * Sets a property. * @param timeout The timeout in ms * @param property The property name. * @param value The property value. */ public void set(int timeout, String property, Object value) { set(timeout, property, value, (SuccessCallback)null); } /** * Sets a property. * @param property The property name. * @param value The property value. */ public void set(String property, Object value) { set(0, property, value); } /** * Sets a property on this javascript object. * @param timeout The timeout in ms. * @param property The property name to set. * @param value The property value. * @param callback Callback which is called when complete */ public void set(int timeout, String property, Object value, SuccessCallback callback) { StringBuilder js = new StringBuilder(); js.append(self+"['"+property+"']=${0}; callback.onSuccess(undefined);"); execute(timeout, js.toString(), new Object[]{value}, callback); } /** * Sets a property on this javascript object. * @param property The property name to set. * @param value The property value. * @param callback Callback which is called when complete */ public void set(String property, Object value, SuccessCallback callback) { set(0, property, value, callback); } /** * Sets multiple properties in a single batch. * @param timeout The timeout in ms * @param properties The properties to set. * @param callback Callback called when operation is completed. */ public void set(int timeout, Map properties, SuccessCallback callback) { StringBuilder js = new StringBuilder(); Object[] params = new Object[properties.size()]; int i=0; for (String key : properties.keySet()) { js.append("self['").append(key).append("']=${").append(i).append("};"); params[i] = properties.get(key); } js.append("callback.onSuccess(undefined)"); execute(timeout, js.toString(), callback); } /** * Sets multiple properties in a single batch. * @param properties The properties to set. * @param callback Callback called when operation is completed. */ public void set(Map properties, SuccessCallback callback) { set(0, properties, callback); } /** * Sets multiple properties in a single batch. * @param timeout Timeout in ms * @param properties The properties to set on this object. */ public void setAndWait(int timeout, Map properties) { StringBuilder js = new StringBuilder(); Object[] params = new Object[properties.size()]; int i=0; for (String key : properties.keySet()) { js.append("self['").append(key).append("']=${").append(i).append("};"); params[i] = properties.get(key); } js.append("callback.onSuccess(undefined)"); executeAndWait(timeout, js.toString()); } /** * Sets multiple properties in a single batch. * @param properties The properties to set on this object. */ public void setAndWait(Map properties) { setAndWait(0, properties); } /** * Sets a property on this javascript object and waits for it to complete using invokeAndBlock. * @param timeout The timeout in ms. * @param property The property name to set. * @param value The value to set. */ public void setAndWait(int timeout, String property, Object value) { StringBuilder js = new StringBuilder(); js.append(self+"."+property+"=${0}; callback.onSuccess(undefined);"); executeAndWait(timeout, js.toString(), new Object[]{value}); } /** * Sets a property on this javascript object and waits for it to complete using invokeAndBlock. * @param property The property name to set. * @param value The value to set. */ public void setAndWait(String property, Object value) { setAndWait(0, property, value); } } /** * A wrapper class for a Javascript value that is returned via the {@link #execute(java.lang.String, com.codename1.util.Callback) } * method. This supports all Javascript primitive types. See {@link JSType} for a list of the types. */ public static class JSRef { /** * The string value of the javascript variable. */ private final String value; /** * The string type of the javascript variable. This is the result returned by the javascript typeof operator. * See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/typeof */ private final String type; /** * Creates a new JSRef object. * @param value The string value of the javascript variable. * @param type The string type of the variable as returned by the typeof operator. */ public JSRef(String value, String type) { this.value = value; this.type = type; } /** * Gets the javascript value as a string. * @return The string value of the reference. */ public String getValue() { return value; } /** * Returns the type of the value * @return */ private String getType() { return type; } /** * Returns the type of the value. * @return */ public JSType getJSType() { return JSType.get(type); } /** * Gets the value as an integer. * @return */ public int getInt() { return new Double(Double.parseDouble(value)).intValue(); } /** * Gets teh value as a double. * @return */ public double getDouble() { return Double.parseDouble(value); } /** * Gets the value as a boolean. * @return */ public boolean getBoolean() { return Boolean.parseBoolean(value); } @Override public String toString() { return value; } /** * Checks if the variable is null * @return */ public boolean isNull() { return value == null; } } /** * Enum with the possible types for a {@link JSRef} object. */ public static enum JSType { OBJECT("object"), FUNCTION("function"), NUMBER("number"), STRING("string"), UNDEFINED("undefined"), BOOLEAN("boolean"); private String typeOfValue; JSType(String val) { typeOfValue = val; } /** * Gets the corresponding JSType for the given string type. * @param type The string type as returned by the typeof operator. Possible input values are 'object', 'function', 'number', 'boolean', and 'undefined' * @return */ public static JSType get(String type) { if ("object".equals(type)) { return OBJECT; } if ("string".equals(type)) { return JSType.STRING; } if ("number".equals(type)) { return JSType.NUMBER; } if ("function".equals(type)) { return JSType.FUNCTION; } if ("undefined".equals(type)) { return JSType.UNDEFINED; } if ("boolean".equals(type)) { return BOOLEAN; } return UNDEFINED; } } /** * Asynchronously executes the provided javascript expression. The expression may provide a callback * which you can call inside the expression directly. * *

Example

* * Getting the {@literal window } object. *
{@code 
     * bc.execute("callback.onSuccess(window)", value -> {
     *    System.out.println("value="+value+"; type="+value.getJSType());
     *       // value=[object Window]; type=OBJECT
     * });
     * }
     * 
*

Getting an Integer

*
{@code 
     * bc.execute("callback.onSuccess(1+2)", value -> {
     *      System.out.println("value="+value.getInt()+"; type="+value.getJSType());
     *          // value=3; type=NUMBER
     * });
     * }
     * 
*

Getting a String

*
{@code 
     * bc.execute("callback.onSuccess('hello world')",value -> {
     *          System.out.println("value="+value+"; type="+value.getJSType());
     *          // value=hello world; type=STRING
     *      }
     * );
     * }
     * 
*

After a Javascript Timeout

*

Since this call is asynchronous, the javascript code can wait to call the * callback to any time in the future - e.g. after a timeout, after an ajax response, * in some event handler, etc.. The CN1 UI will not be blocked, the provided callback * will be called at the appropriate time on the EDT.

*
{@code 
     * bc.execute("setTimeout(function(){callback.onSuccess('hello world')}, 1500)", 
     *      value -> {
     *          System.out.println("value="+value+"; type="+value.getJSType());
     *          // value=hello world; type=STRING
     *      }
     * );
     * }
     * 
* * NOTE: The callback can only be called once, so you shouldn't use this method to register * a callback with an event listener that will be called repeatedly. If you want to register a Java * callback with a Javascript event, you should use the {@link #addJSCallback(java.lang.String, com.codename1.util.Callback) } method * instead. * @param js The javascript expression. If you want to receive any result from this expression, the expression itself must include a call to callback.onSuccess(value). * @param callback The callback. You should call this directly from Javascript. You can call either callback.onSuccess(value) or callback.onError(message,code). */ public void execute(String js, SuccessCallback callback) { StringBuilder fullJs = new StringBuilder(); String isSimulator = Display.getInstance().isSimulator() ? "true":"false"; if (callback == null) { callback = new CallbackAdapter(); } int callbackId = addReturnValueCallback(callback); fullJs .append("(function(){") //.append("cn1application.log('we are here');") .append("var BASE_URL='https://www.codenameone.com").append(RETURN_URL_PREFIX).append("';") .append("function doCallback(val) { ") //.append("cn1application.log('in doCallback');") .append(" var url = BASE_URL + encodeURIComponent(JSON.stringify(val));") .append(" if (window.cefQuery) { window.cefQuery({request:'shouldNavigate:'+url, onSuccess: function(response){}, onFailure:function(error_code, error_message) { console.log(error_message)}});}") .append(" else if (window.cn1application && window.cn1application.shouldNavigate) { window.cn1application.shouldNavigate(url) } else if ("+isSimulator+") {window._cn1ready = window._cn1ready || []; window._cn1ready.push(function(){window.cn1application.shouldNavigate(url)});} else {window.location.href=url}") .append("} ") .append("var result = {value:null, type:null, errorMessage:null, errorCode:0, callbackId:").append(callbackId).append("};") .append("var callback = {") .append(" onSucess: function(val) { this.onSuccess(val);}, ") .append(" onSuccess: function(val) { result.value = val; result.type = typeof(val); if (val !== null && typeof val === 'object') {result.value = val.toString();} doCallback(result);}, ") .append(" onError: function(message, code) { if (message instanceof Error) {result.errorMessage = message.message; result.errorCode = 0;} else {result.errorMessage = message; result.errorCode = code;} doCallback(result);}") .append("};") .append("try { ").append(js).append("} catch (e) {try {callback.onError(e.message, 0);} catch (e2) {callback.onError('Unknown error', 0);}}") .append("})();"); execute(fullJs.toString()); } /** * Execute javascript with a timeout. If timeout is reached before callback is run, * then the callback's onError method is run (if callback is a Callback). If callback isn't a Callback * (i.e. has no onError(), then this will log an error, and call the onSucess method with a null arg. * @param js The javascript to execute * @param timeout The timeout in milliseconds. * @param callback The callback */ public void execute(int timeout, final String js, final SuccessCallback callback) { if (callback != null && timeout > 0) { UITimer.timer(timeout, false, new Runnable() { public void run() { if (returnValueCallbacks().contains(callback)) { Object key = null; for (Map.Entry e : returnValueCallbacks.entrySet()) { if (callback.equals(e.getValue())) { key = e.getKey(); break; } } if (key != null) { if (jsCallbacks == null || !jsCallbacks.contains(callback)) { returnValueCallbacks.remove(key); } if (callback instanceof Callback) { ((Callback)callback).onError(BrowserComponent.this, new RuntimeException("Javascript execution timeout"), 1, "Javascript execution timeout"); } else { Log.e(new RuntimeException("Javascript execution timeout while running "+js)); callback.onSucess(null); } } } } }); } execute(js, callback); } /** * Executes Javascript expression. * @param timeout The timeout in ms * @param js The javascript expression to execute. * @param params Parameters to inject into the javascript expression. The expression should contain placeholders of the form {@literal ${0} }, {@literal ${1} }, etc... to be replaced. See {@link #injectParameters(java.lang.String, java.lang.Object...) } for more information about injected parameters. * by parameters. * @param callback Callback to call when complete. */ public void execute(int timeout, String js, Object[] params, SuccessCallback callback) { execute(0, js, params, callback); } /** * Executes Javascript expression. * @param js The javascript expression to execute. * @param params Parameters to inject into the javascript expression. The expression should contain placeholders of the form {@literal ${0} }, {@literal ${1} }, etc... to be replaced. See {@link #injectParameters(java.lang.String, java.lang.Object...) } for more information about injected parameters. * by parameters. * @param callback Callback to call when complete. */ public void execute(String js, Object[] params, SuccessCallback callback) { execute(injectParameters(js, params), callback); } /** * Sets of callbacks that are registered to persist for multiple calls. */ private Set jsCallbacks; private Set jsCallbacks() { if (jsCallbacks == null) { jsCallbacks = new HashSet(); } return jsCallbacks; } /** * Registers a Java method as a callback in javascript. The {@literal callback} argument * can be referenced inside the javascript expression so that it can be fired when certain events occur. * *

Examples

* *

Register a Callback to be called whenever a button is clicked

* *
     * {@code 
     * bc.addJSCallback("someButton.addEventListener('click', function(){callback.onSuccess('hello world')})", new Callback() {
     *     public void onSucess(JSRef value) {
     *         System.out.println("Received click: "+value);
     *     }
     * });
     * }
     * 
* @param installJs * @param callback */ public void addJSCallback(String installJs, SuccessCallback callback) { jsCallbacks().add(callback); execute(installJs, callback); } /** * Registers Java method as a callback in Javascript. The {@literal callback} argument * can be referenced inside the javascript expression so that it can be fired when certain events occur. * @param installJs The javascript expression. to run. * @param params Parameters to inject into the javascript expression. The expression should contain placeholders of the form {@literal ${0} }, {@literal ${1} }, etc... to be replaced. See {@link #injectParameters(java.lang.String, java.lang.Object...) } for more information about injected parameters. * by parameters. * @param callback The callback to call on completion. */ public void addJSCallback(String installJs, Object[] params, SuccessCallback callback) { addJSCallback(injectParameters(installJs, params), callback); } /** * Removes a JS callback that was added via the {@link #addJSCallback(java.lang.String, com.codename1.util.SuccessCallback) } method. *

Note: This won't unregister any callbacks from the Javascript environment. You'll need to perform your * own additional cleanup in Javascript if this callback is registered in any event handlers. * @param callback The callback to remove. */ public void removeJSCallback(Callback callback) { if (jsCallbacks != null) { jsCallbacks.remove(callback); } } public void removeJSCallback(SuccessCallback callback) { if (jsCallbacks != null) { jsCallbacks.remove(callback); } } private class ExecuteResult { JSRef value; Throwable error; boolean complete; } private static boolean isNumber(Object o) { if (o == null) { return false; } Class c = o.getClass(); return c == Integer.class || c == Double.class || c == Float.class || c == Long.class || c == Short.class; } /** * A wrapper class for a literal javascript expression that can be passed as an * arg in {@link #execute(java.lang.String, java.lang.Object[]) }. */ public static class JSExpression { private String expression; /** * Creates a literal javascript expression. * @param expression The javascript expression. */ public JSExpression(String expression) { this.expression = expression; } /** * Gets the javascript expression as a string. * @return The javascript literal expression. */ public String toString() { return expression; } } /** * Injects parameters into a Javascript string expression. This will quote strings properly. The * expression should include placeholders for each parameter of the form ${0}, ${1}, etc.. * @param jsExpression The javascript expression with placeholders to inject parameters. * @param params * @return The expression with placeholders replaced by parameters. */ public static String injectParameters(String jsExpression, Object... params) { int i=0; for (Object param : params) { String pattern = "${"+i+"}"; if (param == null) { jsExpression = StringUtil.replaceAll(jsExpression, pattern, "null"); } else if (param instanceof String) { jsExpression = StringUtil.replaceAll(jsExpression, pattern, quote((String)param)); } else if (param instanceof JSProxy) { jsExpression = StringUtil.replaceAll(jsExpression, pattern, ((JSProxy)param).self); } else if (param instanceof JSExpression) { jsExpression = ((JSExpression)param).expression; } else if (param instanceof JSRef) { JSRef jsr = (JSRef)param; if (jsr.isNull()) { jsExpression = StringUtil.replaceAll(jsExpression, pattern, "null"); } else if (jsr.getJSType() == JSType.STRING) { jsExpression = StringUtil.replaceAll(jsExpression, pattern, quote((String)jsr.getValue())); } else if (jsr.getJSType() == JSType.FUNCTION || jsr.getJSType() == JSType.OBJECT) { throw new IllegalArgumentException("Cannot inject JSRefs of functions or objects as parameters in JS expressions"); } else { jsExpression = StringUtil.replaceAll(jsExpression, pattern, jsr.getValue()); } } else { jsExpression = StringUtil.replaceAll(jsExpression, pattern, String.valueOf(param)); } i++; } return jsExpression; } /** * Produce a string in double quotes with backslash sequences in all the * right places. A backslash will be inserted within {@link #executeAndWait(java.lang.String)} vs {@link #executeAndReturnString(java.lang.String) } * *

{@link #executeAndReturnString(java.lang.String) } is also blocking, but it uses javascript {@literal eval } * to return the value of the expression. Therefore it can't return the result of any asynchronous operations.

* *

{@link #executeAndWait(java.lang.String) } is built directly on top of {@link #execute(java.lang.String, com.codename1.util.SuccessCallback) } * which is fully asynchronous, and allows you to specify where and when you call the callback within the * javascript code. This means that you must explicitly call either {@code callback.onSuccess(value) } or {@code literalcallback.onError(message, code) } * at some point in the Javascript expression - or the method will block indefinitely.

* @param js The javascript expression to execute. You must call {@code callback.onSuccess(value)} with the result that you want to have returned. * @return The result that is returned from javascript when it calls {@code callback.onSuccess(value) } */ public JSRef executeAndWait(String js) { return executeAndWait(0, js); } /** * This uses invokeAndBlock to wait for the result of the given javascript expression. It is extremely important * that the js expression calls either {@code callback.onSuccess(value) } or {@code literalcallback.onError(message, code) } * at some point, or this method will never return. * *

{@link #executeAndWait(java.lang.String)} vs {@link #executeAndReturnString(java.lang.String) }

* *

{@link #executeAndReturnString(java.lang.String) } is also blocking, but it uses javascript {@literal eval } * to return the value of the expression. Therefore it can't return the result of any asynchronous operations.

* *

{@link #executeAndWait(java.lang.String) } is built directly on top of {@link #execute(java.lang.String, com.codename1.util.SuccessCallback) } * which is fully asynchronous, and allows you to specify where and when you call the callback within the * javascript code. This means that you must explicitly call either {@code callback.onSuccess(value) } or {@code literalcallback.onError(message, code) } * at some point in the Javascript expression - or the method will block indefinitely.

* * @param timeout Timeout in ms * @param js The javascript expression to execute. You must call {@code callback.onSuccess(value)} with the result that you want to have returned. * @return The result that is returned from javascript when it calls {@code callback.onSuccess(value) } */ public JSRef executeAndWait(int timeout, String js) { final ExecuteResult res = new ExecuteResult(); execute(timeout, js, new Callback() { public void onSucess(JSRef value) { synchronized(res) { res.complete = true; res.value = value; res.notifyAll(); } } public void onError(Object sender, Throwable err, int errorCode, String errorMessage) { synchronized(res) { res.complete = true; res.error = err; res.notifyAll(); } } }); while (!res.complete) { Display.getInstance().invokeAndBlock(new Runnable() { public void run() { Util.wait(res, 1000); } }); } if (res.error != null) { throw new RuntimeException(res.error.getMessage()); } else { return res.value; } } /** * Allows exposing the given object to JavaScript code so the JavaScript code can invoke methods * and access fields on the given object. Notice that on RIM devices which don't support reflection * this object must implement the propriatery Scriptable interface * http://www.blackberry.com/developers/docs/5.0.0api/net/rim/device/api/script/Scriptable.html * * @param o the object to invoke, notice all public fields and methods would be exposed to JavaScript * @param name the name to expose within JavaScript * @deprecated this doesn't work in most platforms see issue 459 for details, use the setBrowserNavigationCallback * method instead */ public void exposeInJavaScript(final Object o, final String name) { if (internal == null) { onReady(new Runnable() { public void run() { exposeInJavaScript(o, name); } }); return; } Display.impl.browserExposeInJavaScript(internal, o, name); } /** * Toggles debug mode for the browser component which helps detect coding errors in the JavaScript * bridge logic * @param mode true to debug false otherwise, this might have no effect in some platforms */ public void setDebugMode(boolean mode) { if(mode) { putClientProperty("BrowserComponent.firebug", Boolean.TRUE); } else { putClientProperty("BrowserComponent.firebug", null); } } @Override public void putClientProperty(String key, Object value) { super.putClientProperty(key, value); // In Javascript we use an iframe, and normal behaviour is for the // iframe to be added hidden to the DOM immediately on creation, but // it is removed from the DOM on deinitialize() and added in initComponent(). // In some cases, e.g. WebRTC, removing from the DOM breaks things, so we // need it to remain on the dom even after deinitialize(). This is necessary // in case we reinitialize it afterward (e.g when displaying a dialog, it will // deinitialize the form, and when we close the dialog it will reshow the form // but the browser will be broken. // Thie client property is a flag to tell the JS port not to remove the peer // on deinitialize. if ("HTML5Peer.removeOnDeinitialize".equals(key)) { if (internal != null) { internal.putClientProperty(key, value); } } } /** * Indicates if debug mode is set (might have no effect though) * @return true if debug mode was activated */ public boolean isDebugMode() { return getClientProperty("BrowserComponent.firebug") == Boolean.TRUE; } /** * This method creates a data URI * which allows developers creating HTML for local use to embed local images into the HTML by * appending them as a URI. E.g. instead of referencing a file or URL just load the image data * and place the contents of this string into the src attribute. *

This is the easiest way to get an HTML with local images to work on all mobile platforms. * @param data data of an image * @param mime the mime type of the image e.g. image/png * @return a data URL that can be placed into the img src attribute in HTML e.g.  * @since 6.0 */ public static String createDataURI(byte[] data, String mime) { return "data:" + mime + ";base64," + Base64.encodeNoNewline(data); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy