blob: 90568f66ef49678e0e8892e45ca362a06ce3ef3d [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2003, 2005 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.swt.browser;
import java.io.*;
import org.eclipse.swt.*;
import org.eclipse.swt.widgets.*;
import org.eclipse.swt.graphics.*;
import org.eclipse.swt.internal.*;
import org.eclipse.swt.internal.gtk.GTK;
import org.eclipse.swt.internal.motif.*;
import org.eclipse.swt.internal.mozilla.*;
import org.eclipse.swt.layout.*;
/**
* Instances of this class implement the browser user interface
* metaphor. It allows the user to visualize and navigate through
* HTML documents.
* <p>
* Note that although this class is a subclass of <code>Composite</code>,
* it does not make sense to set a layout on it.
* </p><p>
* IMPORTANT: This class is <em>not</em> intended to be subclassed.
* </p>
*
* @since 3.0
*/
public class Browser extends Composite {
int gtkHandle;
nsIWebBrowser webBrowser;
/* Interfaces for this Mozilla embedding notification */
XPCOMObject supports;
XPCOMObject weakReference;
XPCOMObject webProgressListener;
XPCOMObject webBrowserChrome;
XPCOMObject webBrowserChromeFocus;
XPCOMObject embeddingSiteWindow;
XPCOMObject interfaceRequestor;
XPCOMObject supportsWeakReference;
XPCOMObject contextMenuListener;
XPCOMObject uriContentListener;
XPCOMObject tooltipListener;
int chromeFlags = nsIWebBrowserChrome.CHROME_DEFAULT;
int refCount = 0;
int /*long*/ request;
Point location;
Point size;
boolean addressBar, menuBar, statusBar, toolBar;
boolean visible;
Shell tip = null;
/* External Listener management */
CloseWindowListener[] closeWindowListeners = new CloseWindowListener[0];
LocationListener[] locationListeners = new LocationListener[0];
OpenWindowListener[] openWindowListeners = new OpenWindowListener[0];
ProgressListener[] progressListeners = new ProgressListener[0];
StatusTextListener[] statusTextListeners = new StatusTextListener[0];
TitleListener[] titleListeners = new TitleListener[0];
VisibilityWindowListener[] visibilityWindowListeners = new VisibilityWindowListener[0];
static nsIAppShell AppShell;
static WindowCreator WindowCreator;
static int BrowserCount;
static boolean mozilla;
static boolean IsLinux;
/* Package Name */
static final String PACKAGE_PREFIX = "org.eclipse.swt.browser."; //$NON-NLS-1$
static {
String osName = System.getProperty("os.name").toLowerCase(); //$NON-NLS-1$
IsLinux = osName.startsWith("linux");
}
/**
* Constructs a new instance of this class given its parent
* and a style value describing its behavior and appearance.
* <p>
* The style value is either one of the style constants defined in
* class <code>SWT</code> which is applicable to instances of this
* class, or must be built by <em>bitwise OR</em>'ing together
* (that is, using the <code>int</code> "|" operator) two or more
* of those <code>SWT</code> style constants. The class description
* lists the style constants that are applicable to the class.
* Style bits are also inherited from superclasses.
* </p>
*
* @param parent a widget which will be the parent of the new instance (cannot be null)
* @param style the style of widget to construct
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the parent is null</li>
* </ul>
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the parent</li>
* </ul>
* @exception SWTError <ul>
* <li>ERROR_NO_HANDLES if a handle could not be obtained for browser creation</li>
* </ul>
*
* @see Widget#getStyle
*
* @since 3.0
*/
public Browser(Composite parent, int style) {
super(parent,style | SWT.EMBEDDED);
if (!IsLinux) {
dispose();
SWT.error(SWT.ERROR_NO_HANDLES, null, " [Unsupported platform]"); //$NON-NLS-1$
}
int /*long*/[] result = new int /*long*/[1];
if (!mozilla) {
String mozillaPath = null;
int /*long*/ ptr = OS.getenv(Converter.wcsToMbcs(null, XPCOM.MOZILLA_FIVE_HOME, true));
if (ptr != 0) {
int length = OS.strlen(ptr);
byte[] buffer = new byte[length];
OS.memmove(buffer, ptr, length);
mozillaPath = new String (Converter.mbcsToWcs (null, buffer));
}
if (mozillaPath == null) {
dispose();
SWT.error(SWT.ERROR_NO_HANDLES, null, " [Unknown Mozilla path (MOZILLA_FIVE_HOME not set)]"); //$NON-NLS-1$
}
/*
* Note. Embedding a Mozilla GTK1.2 causes a crash. The workaround
* is to check the version of GTK used by Mozilla by looking for
* the libwidget_gtk.so library used by Mozilla GTK1.2. Mozilla GTK2
* uses the libwidget_gtk2.so library.
*/
File file = new File(mozillaPath, "components/libwidget_gtk.so"); //$NON-NLS-1$
if (file.exists()) {
dispose();
SWT.error(SWT.ERROR_NO_HANDLES, null, " [Mozilla GTK2 required (GTK1.2 detected)]"); //$NON-NLS-1$
}
if (System.getProperty("java.io.tmpdir") == null) { //$NON-NLS-1$
dispose();
SWT.error(SWT.ERROR_NO_HANDLES, null, " [Missing system property java.io.tmpdir is required to create Mozilla profile]"); //$NON-NLS-1$
}
try {
Library.loadLibrary("swt-gtk"); //$NON-NLS-1$
Library.loadLibrary ("swt-mozilla"); //$NON-NLS-1$
} catch (UnsatisfiedLinkError e) {
dispose();
SWT.error(SWT.ERROR_NO_HANDLES, e);
}
int /*long*/[] retVal = new int /*long*/[1];
nsEmbedString path = new nsEmbedString(mozillaPath);
int rc = XPCOM.NS_NewLocalFile(path.getAddress(), true, retVal);
path.dispose();
if (rc != XPCOM.NS_OK) error(rc);
if (retVal[0] == 0) error(XPCOM.NS_ERROR_NULL_POINTER);
nsILocalFile localFile = new nsILocalFile(retVal[0]);
rc = XPCOM.NS_InitEmbedding(localFile.getAddress(), 0);
localFile.Release();
if (rc != XPCOM.NS_OK) {
dispose();
SWT.error(SWT.ERROR_NO_HANDLES, null, " [NS_InitEmbedding "+mozillaPath+" error "+rc+"]"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
}
rc = XPCOM.NS_GetComponentManager(result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
nsIComponentManager componentManager = new nsIComponentManager(result[0]);
result[0] = 0;
rc = componentManager.CreateInstance(XPCOM.NS_APPSHELL_CID, 0, nsIAppShell.NS_IAPPSHELL_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
AppShell = new nsIAppShell(result[0]);
rc = AppShell.Create(0, null);
if (rc != XPCOM.NS_OK) error(rc);
rc = AppShell.Spinup();
if (rc != XPCOM.NS_OK) error(rc);
WindowCreator = new WindowCreator();
WindowCreator.AddRef();
rc = XPCOM.NS_GetServiceManager(result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
nsIServiceManager serviceManager = new nsIServiceManager(result[0]);
result[0] = 0;
byte[] buffer = XPCOM.NS_WINDOWWATCHER_CONTRACTID.getBytes();
byte[] aContractID = new byte[buffer.length + 1];
System.arraycopy(buffer, 0, aContractID, 0, buffer.length);
rc = serviceManager.GetServiceByContractID(aContractID, nsIWindowWatcher.NS_IWINDOWWATCHER_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
serviceManager.Release();
nsIWindowWatcher windowWatcher = new nsIWindowWatcher(result[0]);
result[0] = 0;
rc = windowWatcher.SetWindowCreator(WindowCreator.getAddress());
if (rc != XPCOM.NS_OK) error(rc);
windowWatcher.Release();
PromptServiceFactory factory = new PromptServiceFactory();
factory.AddRef();
rc = componentManager.QueryInterface(nsIComponentRegistrar.NS_ICOMPONENTREGISTRAR_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
nsIComponentRegistrar componentRegistrar = new nsIComponentRegistrar(result[0]);
result[0] = 0;
buffer = XPCOM.NS_PROMPTSERVICE_CONTRACTID.getBytes();
aContractID = new byte[buffer.length + 1];
System.arraycopy(buffer, 0, aContractID, 0, buffer.length);
buffer = "Prompt Service".getBytes(); //$NON-NLS-1$
byte[] aClassName = new byte[buffer.length + 1];
System.arraycopy(buffer, 0, aClassName, 0, buffer.length);
rc = componentRegistrar.RegisterFactory(XPCOM.NS_PROMPTSERVICE_CID, aClassName, aContractID, factory.getAddress());
if (rc != XPCOM.NS_OK) error(rc);
factory.Release();
HelperAppLauncherDialogFactory dialogFactory = new HelperAppLauncherDialogFactory();
dialogFactory.AddRef();
buffer = XPCOM.NS_HELPERAPPLAUNCHERDIALOG_CONTRACTID.getBytes();
aContractID = new byte[buffer.length + 1];
System.arraycopy(buffer, 0, aContractID, 0, buffer.length);
buffer = "Helper App Launcher Dialog".getBytes(); //$NON-NLS-1$
aClassName = new byte[buffer.length + 1];
System.arraycopy(buffer, 0, aClassName, 0, buffer.length);
rc = componentRegistrar.RegisterFactory(XPCOM.NS_HELPERAPPLAUNCHERDIALOG_CID, aClassName, aContractID, dialogFactory.getAddress());
if (rc != XPCOM.NS_OK) error(rc);
dialogFactory.Release();
DownloadFactory downloadFactory = new DownloadFactory();
downloadFactory.AddRef();
buffer = XPCOM.NS_DOWNLOAD_CONTRACTID.getBytes();
aContractID = new byte[buffer.length + 1];
System.arraycopy(buffer, 0, aContractID, 0, buffer.length);
buffer = "Download".getBytes(); //$NON-NLS-1$
aClassName = new byte[buffer.length + 1];
System.arraycopy(buffer, 0, aClassName, 0, buffer.length);
rc = componentRegistrar.RegisterFactory(XPCOM.NS_DOWNLOAD_CID, aClassName, aContractID, downloadFactory.getAddress());
if (rc != XPCOM.NS_OK) error(rc);
downloadFactory.Release();
componentRegistrar.Release();
componentManager.Release();
mozilla = true;
}
BrowserCount++;
if (BrowserCount == 1) {
GTK.gtk_init_check(new int[1], null);
final Display display = getDisplay();
display.asyncExec(new Runnable() {
public void run() {
if (BrowserCount == 0) return;
while (GTK.gtk_events_pending() != 0) {
GTK.gtk_main_iteration();
}
display.timerExec(25, this);
}
});
}
parent.getShell().setFocus();
gtkHandle = GTK.gtk_plug_new(embeddedHandle);
int rc = XPCOM.NS_GetComponentManager(result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
nsIComponentManager componentManager = new nsIComponentManager(result[0]);
result[0] = 0;
nsID NS_IWEBBROWSER_CID = new nsID("F1EAC761-87E9-11d3-AF80-00A024FFC08C"); //$NON-NLS-1$
rc = componentManager.CreateInstance(NS_IWEBBROWSER_CID, 0, nsIWebBrowser.NS_IWEBBROWSER_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
componentManager.Release();
webBrowser = new nsIWebBrowser(result[0]);
createCOMInterfaces();
AddRef();
rc = webBrowser.SetContainerWindow(webBrowserChrome.getAddress());
if (rc != XPCOM.NS_OK) error(rc);
rc = webBrowser.QueryInterface(nsIBaseWindow.NS_IBASEWINDOW_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIBaseWindow baseWindow = new nsIBaseWindow(result[0]);
rc = baseWindow.InitWindow(gtkHandle, 0, 0, 0, 2, 2);
if (rc != XPCOM.NS_OK) error(XPCOM.NS_ERROR_FAILURE);
rc = baseWindow.Create();
if (rc != XPCOM.NS_OK) error(XPCOM.NS_ERROR_FAILURE);
rc = baseWindow.SetVisibility(true);
if (rc != XPCOM.NS_OK) error(XPCOM.NS_ERROR_FAILURE);
baseWindow.Release();
rc = webBrowser.AddWebBrowserListener(weakReference.getAddress(), nsIWebProgressListener.NS_IWEBPROGRESSLISTENER_IID);
if (rc != XPCOM.NS_OK) error(rc);
rc = webBrowser.SetParentURIContentListener(uriContentListener.getAddress());
if (rc != XPCOM.NS_OK) error(rc);
Listener listener = new Listener() {
public void handleEvent(Event event) {
switch (event.type) {
case SWT.Dispose: onDispose(); break;
case SWT.Resize: onResize(); break;
case SWT.FocusIn: Activate(); break;
case SWT.Deactivate: {
Display display = event.display;
if (Browser.this == display.getFocusControl()) Deactivate();
break;
}
}
}
};
int[] folderEvents = new int[]{
SWT.Dispose,
SWT.Resize,
SWT.FocusIn,
SWT.KeyDown,
SWT.Deactivate
};
for (int i = 0; i < folderEvents.length; i++) {
addListener(folderEvents[i], listener);
}
GTK.gtk_widget_show(gtkHandle);
}
/**
* Adds the listener to the collection of listeners who will be
* notified when the window hosting the receiver should be closed.
* <p>
* This notification occurs when a javascript command such as
* <code>window.close</code> gets executed by a <code>Browser</code>.
* </p>
*
* @param listener the listener which should be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void addCloseWindowListener(CloseWindowListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
CloseWindowListener[] newCloseWindowListeners = new CloseWindowListener[closeWindowListeners.length + 1];
System.arraycopy(closeWindowListeners, 0, newCloseWindowListeners, 0, closeWindowListeners.length);
closeWindowListeners = newCloseWindowListeners;
closeWindowListeners[closeWindowListeners.length - 1] = listener;
}
/**
* Adds the listener to the collection of listeners who will be
* notified when the current location has changed or is about to change.
* <p>
* This notification typically occurs when the application navigates
* to a new location with {@link #setUrl(String)} or when the user
* activates a hyperlink.
* </p>
*
* @param listener the listener which should be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void addLocationListener(LocationListener listener) {
checkWidget();
if (listener == null) SWT.error (SWT.ERROR_NULL_ARGUMENT);
LocationListener[] newLocationListeners = new LocationListener[locationListeners.length + 1];
System.arraycopy(locationListeners, 0, newLocationListeners, 0, locationListeners.length);
locationListeners = newLocationListeners;
locationListeners[locationListeners.length - 1] = listener;
}
/**
* Adds the listener to the collection of listeners who will be
* notified when a new window needs to be created.
* <p>
* This notification occurs when a javascript command such as
* <code>window.open</code> gets executed by a <code>Browser</code>.
* </p>
*
* @param listener the listener which should be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void addOpenWindowListener(OpenWindowListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
OpenWindowListener[] newOpenWindowListeners = new OpenWindowListener[openWindowListeners.length + 1];
System.arraycopy(openWindowListeners, 0, newOpenWindowListeners, 0, openWindowListeners.length);
openWindowListeners = newOpenWindowListeners;
openWindowListeners[openWindowListeners.length - 1] = listener;
}
/**
* Adds the listener to the collection of listeners who will be
* notified when a progress is made during the loading of the current
* URL or when the loading of the current URL has been completed.
*
* @param listener the listener which should be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void addProgressListener(ProgressListener listener) {
checkWidget();
if (listener == null) SWT.error (SWT.ERROR_NULL_ARGUMENT);
ProgressListener[] newProgressListeners = new ProgressListener[progressListeners.length + 1];
System.arraycopy(progressListeners, 0, newProgressListeners, 0, progressListeners.length);
progressListeners = newProgressListeners;
progressListeners[progressListeners.length - 1] = listener;
}
/**
* Adds the listener to the collection of listeners who will be
* notified when the status text is changed.
* <p>
* The status text is typically displayed in the status bar of
* a browser application.
* </p>
*
* @param listener the listener which should be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void addStatusTextListener(StatusTextListener listener) {
checkWidget();
if (listener == null) SWT.error (SWT.ERROR_NULL_ARGUMENT);
StatusTextListener[] newStatusTextListeners = new StatusTextListener[statusTextListeners.length + 1];
System.arraycopy(statusTextListeners, 0, newStatusTextListeners, 0, statusTextListeners.length);
statusTextListeners = newStatusTextListeners;
statusTextListeners[statusTextListeners.length - 1] = listener;
}
/**
* Adds the listener to the collection of listeners who will be
* notified when the title of the current document is available
* or has changed.
*
* @param listener the listener which should be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void addTitleListener(TitleListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
TitleListener[] newTitleListeners = new TitleListener[titleListeners.length + 1];
System.arraycopy(titleListeners, 0, newTitleListeners, 0, titleListeners.length);
titleListeners = newTitleListeners;
titleListeners[titleListeners.length - 1] = listener;
}
/**
* Adds the listener to the collection of listeners who will be
* notified when a window hosting the receiver needs to be displayed
* or hidden.
*
* @param listener the listener which should be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void addVisibilityWindowListener(VisibilityWindowListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
VisibilityWindowListener[] newVisibilityWindowListeners = new VisibilityWindowListener[visibilityWindowListeners.length + 1];
System.arraycopy(visibilityWindowListeners, 0, newVisibilityWindowListeners, 0, visibilityWindowListeners.length);
visibilityWindowListeners = newVisibilityWindowListeners;
visibilityWindowListeners[visibilityWindowListeners.length - 1] = listener;
}
/**
* Navigate to the previous session history item.
*
* @return <code>true</code> if the operation was successful and <code>false</code> otherwise
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @see #forward
*
* @since 3.0
*/
public boolean back() {
checkWidget();
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebNavigation.NS_IWEBNAVIGATION_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebNavigation webNavigation = new nsIWebNavigation(result[0]);
rc = webNavigation.GoBack();
webNavigation.Release();
return rc == XPCOM.NS_OK;
}
protected void checkSubclass() {
String name = getClass().getName();
int index = name.lastIndexOf('.');
if (!name.substring(0, index + 1).equals(PACKAGE_PREFIX)) {
SWT.error(SWT.ERROR_INVALID_SUBCLASS);
}
}
void createCOMInterfaces() {
// Create each of the interfaces that this object implements
supports = new XPCOMObject(new int[]{2, 0, 0}){
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
};
weakReference = new XPCOMObject(new int[]{2, 0, 0, 2}){
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return QueryReferent(args[0], args[1]);}
};
webProgressListener = new XPCOMObject(new int[]{2, 0, 0, 4, 6, 3, 4, 3}){
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return OnStateChange(args[0], args[1], args[2],args[3]);}
public int /*long*/ method4(int /*long*/[] args) {return OnProgressChange(args[0], args[1], args[2],args[3],args[4],args[5]);}
public int /*long*/ method5(int /*long*/[] args) {return OnLocationChange(args[0], args[1], args[2]);}
public int /*long*/ method6(int /*long*/[] args) {return OnStatusChange(args[0], args[1], args[2],args[3]);}
public int /*long*/ method7(int /*long*/[] args) {return OnSecurityChange(args[0], args[1], args[2]);}
};
webBrowserChrome = new XPCOMObject(new int[]{2, 0, 0, 2, 1, 1, 1, 1, 0, 2, 0, 1, 1}){
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return SetStatus(args[0], args[1]);}
public int /*long*/ method4(int /*long*/[] args) {return GetWebBrowser(args[0]);}
public int /*long*/ method5(int /*long*/[] args) {return SetWebBrowser(args[0]);}
public int /*long*/ method6(int /*long*/[] args) {return GetChromeFlags(args[0]);}
public int /*long*/ method7(int /*long*/[] args) {return SetChromeFlags(args[0]);}
public int /*long*/ method8(int /*long*/[] args) {return DestroyBrowserWindow();}
public int /*long*/ method9(int /*long*/[] args) {return SizeBrowserTo(args[0], args[1]);}
public int /*long*/ method10(int /*long*/[] args) {return ShowAsModal();}
public int /*long*/ method11(int /*long*/[] args) {return IsWindowModal(args[0]);}
public int /*long*/ method12(int /*long*/[] args) {return ExitModalEventLoop(args[0]);}
};
webBrowserChromeFocus = new XPCOMObject(new int[]{2, 0, 0, 0, 0}){
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return FocusNextElement();}
public int /*long*/ method4(int /*long*/[] args) {return FocusPrevElement();}
};
embeddingSiteWindow = new XPCOMObject(new int[]{2, 0, 0, 5, 5, 0, 1, 1, 1, 1, 1}){
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return SetDimensions(args[0], args[1], args[2], args[3], args[4]);}
public int /*long*/ method4(int /*long*/[] args) {return GetDimensions(args[0], args[1], args[2], args[3], args[4]);}
public int /*long*/ method5(int /*long*/[] args) {return SetFocus();}
public int /*long*/ method6(int /*long*/[] args) {return GetVisibility(args[0]);}
public int /*long*/ method7(int /*long*/[] args) {return SetVisibility(args[0]);}
public int /*long*/ method8(int /*long*/[] args) {return GetTitle(args[0]);}
public int /*long*/ method9(int /*long*/[] args) {return SetTitle(args[0]);}
public int /*long*/ method10(int /*long*/[] args) {return GetSiteWindow(args[0]);}
};
interfaceRequestor = new XPCOMObject(new int[]{2, 0, 0, 2}){
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return GetInterface(args[0], args[1]);}
};
supportsWeakReference = new XPCOMObject(new int[]{2, 0, 0, 1}){
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return GetWeakReference(args[0]);}
};
contextMenuListener = new XPCOMObject(new int[]{2, 0, 0, 3}){
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return OnShowContextMenu(args[0],args[1],args[2]);}
};
uriContentListener = new XPCOMObject(new int[]{2, 0, 0, 2, 5, 3, 4, 1, 1, 1, 1}) {
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return OnStartURIOpen(args[0], args[1]);}
public int /*long*/ method4(int /*long*/[] args) {return DoContent(args[0], args[1], args[2], args[3], args[4]);}
public int /*long*/ method5(int /*long*/[] args) {return IsPreferred(args[0], args[1], args[2]);}
public int /*long*/ method6(int /*long*/[] args) {return CanHandleContent(args[0], args[1], args[2], args[3]);}
public int /*long*/ method7(int /*long*/[] args) {return GetLoadCookie(args[0]);}
public int /*long*/ method8(int /*long*/[] args) {return SetLoadCookie(args[0]);}
public int /*long*/ method9(int /*long*/[] args) {return GetParentContentListener(args[0]);}
public int /*long*/ method10(int /*long*/[] args) {return SetParentContentListener(args[0]);}
};
tooltipListener = new XPCOMObject(new int[]{2, 0, 0, 3, 0}) {
public int /*long*/ method0(int /*long*/[] args) {return QueryInterface(args[0], args[1]);}
public int /*long*/ method1(int /*long*/[] args) {return AddRef();}
public int /*long*/ method2(int /*long*/[] args) {return Release();}
public int /*long*/ method3(int /*long*/[] args) {return OnShowTooltip(args[0], args[1], args[2]);}
public int /*long*/ method4(int /*long*/[] args) {return OnHideTooltip();}
};
}
void disposeCOMInterfaces() {
if (supports != null) {
supports.dispose();
supports = null;
}
if (weakReference != null) {
weakReference.dispose();
weakReference = null;
}
if (webProgressListener != null) {
webProgressListener.dispose();
webProgressListener = null;
}
if (webBrowserChrome != null) {
webBrowserChrome.dispose();
webBrowserChrome = null;
}
if (webBrowserChromeFocus != null) {
webBrowserChromeFocus.dispose();
webBrowserChromeFocus = null;
}
if (embeddingSiteWindow != null) {
embeddingSiteWindow.dispose();
embeddingSiteWindow = null;
}
if (interfaceRequestor != null) {
interfaceRequestor.dispose();
interfaceRequestor = null;
}
if (supportsWeakReference != null) {
supportsWeakReference.dispose();
supportsWeakReference = null;
}
if (contextMenuListener != null) {
contextMenuListener.dispose();
contextMenuListener = null;
}
if (uriContentListener != null) {
uriContentListener.dispose();
uriContentListener = null;
}
if (tooltipListener != null) {
tooltipListener.dispose();
tooltipListener = null;
}
}
/**
* Execute the specified script.
*
* <p>
* Execute a script containing javascript commands in the context of the current document.
*
* @param script the script with javascript commands
*
* @return <code>true</code> if the operation was successful and <code>false</code> otherwise
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the script is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.1
*/
public boolean execute(String script) {
checkWidget();
if (script == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
String url = "javascript:"+script+";void(0);";//$NON-NLS-1$ //$NON-NLS-2$
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebNavigation.NS_IWEBNAVIGATION_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebNavigation webNavigation = new nsIWebNavigation(result[0]);
char[] arg = url.toCharArray();
char[] c = new char[arg.length+1];
System.arraycopy(arg,0,c,0,arg.length);
rc = webNavigation.LoadURI(c, nsIWebNavigation.LOAD_FLAGS_NONE, 0, 0, 0);
webNavigation.Release();
return rc == XPCOM.NS_OK;
}
static Browser findBrowser(int /*long*/ handle) {
Display display = Display.getCurrent();
Shell[] shells = display.getShells();
Browser browser = null;
for (int i = 0; i < shells.length; i++) {
browser = Browser.findBrowser(shells[i], handle);
if (browser != null) break;
}
return browser;
}
static Browser findBrowser(Control control, int gtkHandle) {
if (control instanceof Browser) {
Browser browser = (Browser)control;
if (browser.gtkHandle == gtkHandle) return browser;
}
if (control instanceof Composite) {
Composite composite = (Composite)control;
Control[] children = composite.getChildren();
for (int i = 0; i < children.length; i++) {
Browser browser = findBrowser(children[i], gtkHandle);
if (browser != null) return browser;
}
}
return null;
}
/**
* Navigate to the next session history item.
*
* @return <code>true</code> if the operation was successful and <code>false</code> otherwise
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @see #back
*
* @since 3.0
*/
public boolean forward() {
checkWidget();
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebNavigation.NS_IWEBNAVIGATION_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebNavigation webNavigation = new nsIWebNavigation(result[0]);
rc = webNavigation.GoForward();
webNavigation.Release();
return rc == XPCOM.NS_OK;
}
/**
* Returns the current URL.
*
* @return the current URL or an empty <code>String</code> if there is no current URL
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @see #setUrl
*
* @since 3.0
*/
public String getUrl() {
checkWidget();
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebNavigation.NS_IWEBNAVIGATION_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebNavigation webNavigation = new nsIWebNavigation(result[0]);
int /*long*/[] aCurrentURI = new int /*long*/[1];
rc = webNavigation.GetCurrentURI(aCurrentURI);
if (rc != XPCOM.NS_OK) error(rc);
/*
* This code is intentionally commented. aCurrentURI is 0
* when no location has previously been set.
*/
//if (aCurrentURI[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
webNavigation.Release();
byte[] dest = null;
if (aCurrentURI[0] != 0) {
nsIURI uri = new nsIURI(aCurrentURI[0]);
int /*long*/ aSpec = XPCOM.nsEmbedCString_new();
rc = uri.GetSpec(aSpec);
if (rc != XPCOM.NS_OK) error(rc);
int length = XPCOM.nsEmbedCString_Length(aSpec);
int /*long*/ buffer = XPCOM.nsEmbedCString_get(aSpec);
dest = new byte[length];
XPCOM.memmove(dest, buffer, length);
XPCOM.nsEmbedCString_delete(aSpec);
uri.Release();
}
return dest != null ? new String(dest) : ""; //$NON-NLS-1$
}
/**
* Returns <code>true</code> if the receiver can navigate to the
* previous session history item, and <code>false</code> otherwise.
*
* @return the receiver's back command enabled state
*
* @exception SWTException <ul>
* <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
* <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
* </ul>
*
* @see #back
*/
public boolean isBackEnabled() {
checkWidget();
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebNavigation.NS_IWEBNAVIGATION_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebNavigation webNavigation = new nsIWebNavigation(result[0]);
boolean[] aCanGoBack = new boolean[1];
rc = webNavigation.GetCanGoBack(aCanGoBack);
webNavigation.Release();
return aCanGoBack[0];
}
/**
* Returns <code>true</code> if the receiver can navigate to the
* next session history item, and <code>false</code> otherwise.
*
* @return the receiver's forward command enabled state
*
* @exception SWTException <ul>
* <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
* <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
* </ul>
*
* @see #forward
*/
public boolean isForwardEnabled() {
checkWidget();
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebNavigation.NS_IWEBNAVIGATION_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebNavigation webNavigation = new nsIWebNavigation(result[0]);
boolean[] aCanGoForward = new boolean[1];
rc = webNavigation.GetCanGoForward(aCanGoForward);
webNavigation.Release();
return aCanGoForward[0];
}
static String error(int code) {
throw new SWTError("XPCOM error "+code); //$NON-NLS-1$
}
void onDispose() {
int rc = webBrowser.RemoveWebBrowserListener(weakReference.getAddress(), nsIWebProgressListener.NS_IWEBPROGRESSLISTENER_IID);
if (rc != XPCOM.NS_OK) error(rc);
rc = webBrowser.SetParentURIContentListener(0);
if (rc != XPCOM.NS_OK) error(rc);
int /*long*/[] result = new int /*long*/[1];
rc = webBrowser.QueryInterface(nsIBaseWindow.NS_IBASEWINDOW_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIBaseWindow baseWindow = new nsIBaseWindow(result[0]);
rc = baseWindow.Destroy();
if (rc != XPCOM.NS_OK) error(rc);
baseWindow.Release();
Release();
webBrowser.Release();
if (tip != null && !tip.isDisposed()) tip.dispose();
tip = null;
GTK.gtk_widget_destroy(gtkHandle);
while (GTK.gtk_events_pending() != 0) {
GTK.gtk_main_iteration();
}
gtkHandle = 0;
BrowserCount--;
/*
* This code is intentionally commented. It is not possible to reinitialize
* Mozilla once it has been terminated. NS_InitEmbedding always fails after
* NS_TermEmbedding has been called. The workaround is to call NS_InitEmbedding
* once and never call NS_TermEmbedding.
*/
// if (BrowserCount == 0) {
// if (AppShell != null) {
// // Shutdown the appshell service.
// rc = AppShell.Spindown();
// if (rc != XPCOM.NS_OK) error(rc);
// AppShell.Release();
// AppShell = null;
// }
// WindowCreator.Release();
// WindowCreator = null;
// PromptService.Release();
// PromptService = null;
// XPCOM.NS_TermEmbedding();
// mozilla = false;
// }
}
void Activate() {
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebBrowserFocus.NS_IWEBBROWSERFOCUS_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebBrowserFocus webBrowserFocus = new nsIWebBrowserFocus(result[0]);
rc = webBrowserFocus.Activate();
if (rc != XPCOM.NS_OK) error(rc);
webBrowserFocus.Release();
}
void Deactivate() {
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebBrowserFocus.NS_IWEBBROWSERFOCUS_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebBrowserFocus webBrowserFocus = new nsIWebBrowserFocus(result[0]);
rc = webBrowserFocus.Deactivate();
if (rc != XPCOM.NS_OK) error(rc);
webBrowserFocus.Release();
}
void SetFocusAtFirstElement() {
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebBrowserFocus.NS_IWEBBROWSERFOCUS_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebBrowserFocus webBrowserFocus = new nsIWebBrowserFocus(result[0]);
rc = webBrowserFocus.SetFocusAtFirstElement();
if (rc != XPCOM.NS_OK) error(rc);
webBrowserFocus.Release();
}
void onResize() {
Rectangle rect = getClientArea();
int width = Math.max(2, rect.width);
int height = Math.max(2, rect.height);
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIBaseWindow.NS_IBASEWINDOW_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIBaseWindow baseWindow = new nsIBaseWindow(result[0]);
rc = baseWindow.SetPositionAndSize(0, 0, width, height, true);
if (rc != XPCOM.NS_OK) error(rc);
baseWindow.Release();
}
/**
* Refresh the current page.
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void refresh() {
checkWidget();
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebNavigation.NS_IWEBNAVIGATION_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebNavigation webNavigation = new nsIWebNavigation(result[0]);
rc = webNavigation.Reload(nsIWebNavigation.LOAD_FLAGS_NONE);
/*
* Feature in Mozilla. Reload returns an error code NS_ERROR_INVALID_POINTER
* when it is called immediately after a request to load a new document using
* LoadURI. The workaround is to ignore this error code.
*/
if (rc != XPCOM.NS_OK && rc != XPCOM.NS_ERROR_INVALID_POINTER) error(rc);
webNavigation.Release();
}
/**
* Removes the listener from the collection of listeners who will
* be notified when the window hosting the receiver should be closed.
*
* @param listener the listener which should no longer be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void removeCloseWindowListener(CloseWindowListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
if (closeWindowListeners.length == 0) return;
int index = -1;
for (int i = 0; i < closeWindowListeners.length; i++) {
if (listener == closeWindowListeners[i]){
index = i;
break;
}
}
if (index == -1) return;
if (closeWindowListeners.length == 1) {
closeWindowListeners = new CloseWindowListener[0];
return;
}
CloseWindowListener[] newCloseWindowListeners = new CloseWindowListener[closeWindowListeners.length - 1];
System.arraycopy(closeWindowListeners, 0, newCloseWindowListeners, 0, index);
System.arraycopy(closeWindowListeners, index + 1, newCloseWindowListeners, index, closeWindowListeners.length - index - 1);
closeWindowListeners = newCloseWindowListeners;
}
/**
* Removes the listener from the collection of listeners who will
* be notified when the current location is changed or about to be changed.
*
* @param listener the listener which should no longer be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void removeLocationListener(LocationListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
if (locationListeners.length == 0) return;
int index = -1;
for (int i = 0; i < locationListeners.length; i++) {
if (listener == locationListeners[i]){
index = i;
break;
}
}
if (index == -1) return;
if (locationListeners.length == 1) {
locationListeners = new LocationListener[0];
return;
}
LocationListener[] newLocationListeners = new LocationListener[locationListeners.length - 1];
System.arraycopy(locationListeners, 0, newLocationListeners, 0, index);
System.arraycopy(locationListeners, index + 1, newLocationListeners, index, locationListeners.length - index - 1);
locationListeners = newLocationListeners;
}
/**
* Removes the listener from the collection of listeners who will
* be notified when a new window needs to be created.
*
* @param listener the listener which should no longer be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void removeOpenWindowListener(OpenWindowListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
if (openWindowListeners.length == 0) return;
int index = -1;
for (int i = 0; i < openWindowListeners.length; i++) {
if (listener == openWindowListeners[i]){
index = i;
break;
}
}
if (index == -1) return;
if (openWindowListeners.length == 1) {
openWindowListeners = new OpenWindowListener[0];
return;
}
OpenWindowListener[] newOpenWindowListeners = new OpenWindowListener[openWindowListeners.length - 1];
System.arraycopy(openWindowListeners, 0, newOpenWindowListeners, 0, index);
System.arraycopy(openWindowListeners, index + 1, newOpenWindowListeners, index, openWindowListeners.length - index - 1);
openWindowListeners = newOpenWindowListeners;
}
/**
* Removes the listener from the collection of listeners who will
* be notified when a progress is made during the loading of the current
* URL or when the loading of the current URL has been completed.
*
* @param listener the listener which should no longer be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void removeProgressListener(ProgressListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
if (progressListeners.length == 0) return;
int index = -1;
for (int i = 0; i < progressListeners.length; i++) {
if (listener == progressListeners[i]){
index = i;
break;
}
}
if (index == -1) return;
if (progressListeners.length == 1) {
progressListeners = new ProgressListener[0];
return;
}
ProgressListener[] newProgressListeners = new ProgressListener[progressListeners.length - 1];
System.arraycopy(progressListeners, 0, newProgressListeners, 0, index);
System.arraycopy(progressListeners, index + 1, newProgressListeners, index, progressListeners.length - index - 1);
progressListeners = newProgressListeners;
}
/**
* Removes the listener from the collection of listeners who will
* be notified when the status text is changed.
*
* @param listener the listener which should no longer be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void removeStatusTextListener(StatusTextListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
if (statusTextListeners.length == 0) return;
int index = -1;
for (int i = 0; i < statusTextListeners.length; i++) {
if (listener == statusTextListeners[i]){
index = i;
break;
}
}
if (index == -1) return;
if (statusTextListeners.length == 1) {
statusTextListeners = new StatusTextListener[0];
return;
}
StatusTextListener[] newStatusTextListeners = new StatusTextListener[statusTextListeners.length - 1];
System.arraycopy(statusTextListeners, 0, newStatusTextListeners, 0, index);
System.arraycopy(statusTextListeners, index + 1, newStatusTextListeners, index, statusTextListeners.length - index - 1);
statusTextListeners = newStatusTextListeners;
}
/**
* Removes the listener from the collection of listeners who will
* be notified when the title of the current document is available
* or has changed.
*
* @param listener the listener which should no longer be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void removeTitleListener(TitleListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
if (titleListeners.length == 0) return;
int index = -1;
for (int i = 0; i < titleListeners.length; i++) {
if (listener == titleListeners[i]){
index = i;
break;
}
}
if (index == -1) return;
if (titleListeners.length == 1) {
titleListeners = new TitleListener[0];
return;
}
TitleListener[] newTitleListeners = new TitleListener[titleListeners.length - 1];
System.arraycopy(titleListeners, 0, newTitleListeners, 0, index);
System.arraycopy(titleListeners, index + 1, newTitleListeners, index, titleListeners.length - index - 1);
titleListeners = newTitleListeners;
}
/**
* Removes the listener from the collection of listeners who will
* be notified when a window hosting the receiver needs to be displayed
* or hidden.
*
* @param listener the listener which should no longer be notified
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void removeVisibilityWindowListener(VisibilityWindowListener listener) {
checkWidget();
if (listener == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
if (visibilityWindowListeners.length == 0) return;
int index = -1;
for (int i = 0; i < visibilityWindowListeners.length; i++) {
if (listener == visibilityWindowListeners[i]){
index = i;
break;
}
}
if (index == -1) return;
if (visibilityWindowListeners.length == 1) {
visibilityWindowListeners = new VisibilityWindowListener[0];
return;
}
VisibilityWindowListener[] newVisibilityWindowListeners = new VisibilityWindowListener[visibilityWindowListeners.length - 1];
System.arraycopy(visibilityWindowListeners, 0, newVisibilityWindowListeners, 0, index);
System.arraycopy(visibilityWindowListeners, index + 1, newVisibilityWindowListeners, index, visibilityWindowListeners.length - index - 1);
visibilityWindowListeners = newVisibilityWindowListeners;
}
/**
* Renders HTML.
*
* <p>
* The html parameter is Unicode encoded since it is a java <code>String</code>.
* As a result, the HTML meta tag charset should not be set. The charset is implied
* by the <code>String</code> itself.
*
* @param html the HTML content to be rendered
*
* @return true if the operation was successful and false otherwise.
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the html is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @see #setUrl
*
* @since 3.0
*/
public boolean setText(String html) {
checkWidget();
if (html == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
/*
* Feature in Mozilla. The focus memory of Mozilla must be
* properly managed through the nsIWebBrowserFocus interface.
* In particular, nsIWebBrowserFocus.deactivate must be called
* when the focus moves from the browser (or one of its children
* managed by Mozilla to another widget. We currently do not
* get notified when a widget takes focus away from the Browser.
* As a result, deactivate is not properly called. This causes
* Mozilla to retake focus the next time a document is loaded.
* This breaks the case where the HTML loaded in the Browser
* varies while the user enters characters in a text widget. The text
* widget loses focus every time new content is loaded.
* The current workaround is to call deactivate everytime if
* the browser currently does not have focus. A better workaround
* would be to have a mean to call deactivate when the Browser
* or one of its children loses focus.
*/
if (this != getDisplay().getFocusControl()) Deactivate();
/* Convert the String containing HTML to an array of
* bytes with UTF-8 data.
*/
byte[] data = null;
try {
data = html.getBytes("UTF-8"); //$NON-NLS-1$
} catch (UnsupportedEncodingException e) {
return false;
}
/* render HTML in memory */
InputStream inputStream = new InputStream(data);
inputStream.AddRef();
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIInterfaceRequestor.NS_IINTERFACEREQUESTOR_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIInterfaceRequestor interfaceRequestor = new nsIInterfaceRequestor(result[0]);
result[0] = 0;
rc = interfaceRequestor.GetInterface(nsIDocShell.NS_IDOCSHELL_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
interfaceRequestor.Release();
nsIDocShell docShell = new nsIDocShell(result[0]);
result[0] = 0;
rc = XPCOM.NS_GetServiceManager(result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
nsIServiceManager serviceManager = new nsIServiceManager(result[0]);
result[0] = 0;
rc = serviceManager.GetService(XPCOM.NS_IOSERVICE_CID, nsIIOService.NS_IIOSERVICE_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
nsIIOService ioService = new nsIIOService(result[0]);
result[0] = 0;
/*
* Note. Mozilla ignores LINK tags used to load CSS stylesheets
* when the URI protocol for the nsInputStreamChannel
* is about:blank. The fix is to specify the file protocol.
*/
byte[] aString = "file:".getBytes(); //$NON-NLS-1$
int /*long*/ aSpec = XPCOM.nsEmbedCString_new(aString, aString.length);
rc = ioService.NewURI(aSpec, null, 0, result);
XPCOM.nsEmbedCString_delete(aSpec);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
ioService.Release();
nsIURI uri = new nsIURI(result[0]);
result[0] = 0;
rc = XPCOM.NS_GetComponentManager(result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
/* aContentType */
byte[] buffer = "text/plain".getBytes(); //$NON-NLS-1$
byte[] contentTypeBuffer = new byte[buffer.length + 1];
System.arraycopy(buffer, 0, contentTypeBuffer, 0, buffer.length);
int /*long*/ aContentType = XPCOM.nsEmbedCString_new(contentTypeBuffer, contentTypeBuffer.length);
buffer = "UTF-8".getBytes(); //$NON-NLS-1$
byte[] contentCharsetBuffer = new byte[buffer.length + 1];
System.arraycopy(buffer, 0, contentCharsetBuffer, 0, buffer.length);
int /*long*/ aContentCharset = XPCOM.nsEmbedCString_new(contentCharsetBuffer, contentCharsetBuffer.length);
/*
* Feature in Mozilla. LoadStream invokes the nsIInputStream argument
* through a different thread. The callback mechanism must attach
* a non java thread to the JVM otherwise the nsIInputStream Read and
* Close methods never get called.
*/
rc = docShell.LoadStream(inputStream.getAddress(), uri.getAddress(), aContentType, aContentCharset, 0);
if (rc != XPCOM.NS_OK) error(rc);
XPCOM.nsEmbedCString_delete(aContentCharset);
XPCOM.nsEmbedCString_delete(aContentType);
uri.Release();
inputStream.Release();
docShell.Release();
return true;
}
/**
* Loads a URL.
*
* @param url the URL to be loaded
*
* @return true if the operation was successful and false otherwise.
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT - if the url is null</li>
* </ul>
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @see #getUrl
*
* @since 3.0
*/
public boolean setUrl(String url) {
checkWidget();
if (url == null) SWT.error(SWT.ERROR_NULL_ARGUMENT);
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebNavigation.NS_IWEBNAVIGATION_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebNavigation webNavigation = new nsIWebNavigation(result[0]);
char[] arg = url.toCharArray();
char[] c = new char[arg.length+1];
System.arraycopy(arg,0,c,0,arg.length);
rc = webNavigation.LoadURI(c, nsIWebNavigation.LOAD_FLAGS_NONE, 0, 0, 0);
webNavigation.Release();
return rc == XPCOM.NS_OK;
}
/**
* Stop any loading and rendering activity.
*
* @exception SWTException <ul>
* <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread</li>
* <li>ERROR_WIDGET_DISPOSED when the widget has been disposed</li>
* </ul>
*
* @since 3.0
*/
public void stop() {
checkWidget();
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIWebNavigation.NS_IWEBNAVIGATION_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIWebNavigation webNavigation = new nsIWebNavigation(result[0]);
rc = webNavigation.Stop(nsIWebNavigation.STOP_ALL);
if (rc != XPCOM.NS_OK) error(rc);
webNavigation.Release();
}
/* nsISupports */
int /*long*/ QueryInterface(int /*long*/ riid, int /*long*/ ppvObject) {
if (riid == 0 || ppvObject == 0) return XPCOM.NS_ERROR_NO_INTERFACE;
nsID guid = new nsID();
XPCOM.memmove(guid, riid, nsID.sizeof);
if (guid.Equals(nsISupports.NS_ISUPPORTS_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {supports.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsIWeakReference.NS_IWEAKREFERENCE_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {weakReference.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsIWebProgressListener.NS_IWEBPROGRESSLISTENER_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {webProgressListener.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsIWebBrowserChrome.NS_IWEBBROWSERCHROME_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {webBrowserChrome.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsIWebBrowserChromeFocus.NS_IWEBBROWSERCHROMEFOCUS_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {webBrowserChromeFocus.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsIEmbeddingSiteWindow.NS_IEMBEDDINGSITEWINDOW_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {embeddingSiteWindow.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsIInterfaceRequestor.NS_IINTERFACEREQUESTOR_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {interfaceRequestor.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsISupportsWeakReference.NS_ISUPPORTSWEAKREFERENCE_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {supportsWeakReference.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsIContextMenuListener.NS_ICONTEXTMENULISTENER_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {contextMenuListener.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsIURIContentListener.NS_IURICONTENTLISTENER_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {uriContentListener.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
if (guid.Equals(nsITooltipListener.NS_ITOOLTIPLISTENER_IID)) {
XPCOM.memmove(ppvObject, new int /*long*/[] {tooltipListener.getAddress()}, OS.PTR_SIZEOF);
AddRef();
return XPCOM.NS_OK;
}
XPCOM.memmove(ppvObject, new int /*long*/[] {0}, 4);
return XPCOM.NS_ERROR_NO_INTERFACE;
}
int /*long*/ AddRef() {
refCount++;
return refCount;
}
int /*long*/ Release() {
refCount--;
if (refCount == 0) disposeCOMInterfaces();
return refCount;
}
/* nsIWeakReference */
int /*long*/ QueryReferent(int /*long*/ riid, int /*long*/ ppvObject) {
return QueryInterface(riid,ppvObject);
}
/* nsIInterfaceRequestor */
int /*long*/ GetInterface(int /*long*/ riid, int /*long*/ ppvObject) {
if (riid == 0 || ppvObject == 0) return XPCOM.NS_ERROR_NO_INTERFACE;
nsID guid = new nsID();
XPCOM.memmove(guid, riid, nsID.sizeof);
if (guid.Equals(nsIDOMWindow.NS_IDOMWINDOW_IID)) {
int /*long*/[] aContentDOMWindow = new int /*long*/[1];
int rc = webBrowser.GetContentDOMWindow(aContentDOMWindow);
if (rc != XPCOM.NS_OK) error(rc);
if (aContentDOMWindow[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
XPCOM.memmove(ppvObject, aContentDOMWindow, 4);
return rc;
}
return QueryInterface(riid,ppvObject);
}
int /*long*/ GetWeakReference(int /*long*/ ppvObject) {
XPCOM.memmove(ppvObject, new int /*long*/[] {weakReference.getAddress()}, 4);
AddRef();
return XPCOM.NS_OK;
}
/* nsIWebProgressListener */
int /*long*/ OnStateChange(int /*long*/ aWebProgress, int /*long*/ aRequest, int /*long*/ aStateFlags, int /*long*/ aStatus) {
if ((aStateFlags & nsIWebProgressListener.STATE_IS_DOCUMENT) == 0) return XPCOM.NS_OK;
if ((aStateFlags & nsIWebProgressListener.STATE_START) != 0) {
if (request == 0) request = aRequest;
} else if ((aStateFlags & nsIWebProgressListener.STATE_REDIRECTING) != 0) {
if (request == aRequest) request = 0;
} else if ((aStateFlags & nsIWebProgressListener.STATE_STOP) != 0) {
/*
* Feature on Mozilla. When a request is redirected (STATE_REDIRECTING),
* it never reaches the state STATE_STOP and it is replaced with a new request.
* The new request is received when it is in the state STATE_STOP.
* To handle this case, the variable request is set to 0 when the corresponding
* request is redirected. The following request received with the state STATE_STOP
* - the new request resulting from the redirection - is used to send
* the ProgressListener.completed event.
*/
if (request == aRequest || request == 0) {
request = 0;
StatusTextEvent event = new StatusTextEvent(this);
event.display = getDisplay();
event.widget = this;
event.text = ""; //$NON-NLS-1$
for (int i = 0; i < statusTextListeners.length; i++)
statusTextListeners[i].changed(event);
ProgressEvent event2 = new ProgressEvent(this);
event2.display = getDisplay();
event2.widget = this;
for (int i = 0; i < progressListeners.length; i++)
progressListeners[i].completed(event2);
}
}
return XPCOM.NS_OK;
}
int /*long*/ OnProgressChange(int /*long*/ aWebProgress, int /*long*/ aRequest, int /*long*/ aCurSelfProgress, int /*long*/ aMaxSelfProgress, int /*long*/ aCurTotalProgress, int /*long*/ aMaxTotalProgress) {
if (progressListeners.length == 0) return XPCOM.NS_OK;
int /*long*/ total = aMaxTotalProgress;
if (total <= 0) total = Integer.MAX_VALUE;
ProgressEvent event = new ProgressEvent(this);
event.display = getDisplay();
event.widget = this;
event.current = (int)/*64*/aCurTotalProgress;
event.total = (int)/*64*/aMaxTotalProgress;
for (int i = 0; i < progressListeners.length; i++)
progressListeners[i].changed(event);
return XPCOM.NS_OK;
}
int /*long*/ OnLocationChange(int /*long*/ aWebProgress, int /*long*/ aRequest, int /*long*/ aLocation) {
/*
* Feature on Mozilla. When a page is loaded via setText before a previous
* setText page load has completed, the expected OnStateChange STATE_STOP for the
* original setText never arrives because it gets replaced by the OnStateChange
* STATE_STOP for the new request. This results in the request field never being
* cleared because the original request's OnStateChange STATE_STOP is still expected
* (but never arrives). To handle this case, the request field is updated to the new
* overriding request since its OnStateChange STATE_STOP will be received next.
*/
if (request != 0 && request != aRequest) request = aRequest;
if (locationListeners.length == 0) return XPCOM.NS_OK;
nsIWebProgress webProgress = new nsIWebProgress(aWebProgress);
int /*long*/[] aDOMWindow = new int /*long*/[1];
int rc = webProgress.GetDOMWindow(aDOMWindow);
if (rc != XPCOM.NS_OK) error(rc);
if (aDOMWindow[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIDOMWindow domWindow = new nsIDOMWindow(aDOMWindow[0]);
int /*long*/[] aTop = new int /*long*/[1];
rc = domWindow.GetTop(aTop);
if (rc != XPCOM.NS_OK) error(rc);
if (aTop[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
domWindow.Release();
nsIDOMWindow topWindow = new nsIDOMWindow(aTop[0]);
topWindow.Release();
nsIURI location = new nsIURI(aLocation);
int /*long*/ aSpec = XPCOM.nsEmbedCString_new();
location.GetSpec(aSpec);
int length = XPCOM.nsEmbedCString_Length(aSpec);
int /*long*/ buffer = XPCOM.nsEmbedCString_get(aSpec);
byte[] dest = new byte[length];
XPCOM.memmove(dest, buffer, length);
XPCOM.nsEmbedCString_delete(aSpec);
LocationEvent event = new LocationEvent(this);
event.display = getDisplay();
event.widget = this;
event.location = new String(dest);
event.top = aTop[0] == aDOMWindow[0];
for (int i = 0; i < locationListeners.length; i++)
locationListeners[i].changed(event);
return XPCOM.NS_OK;
}
int /*long*/ OnStatusChange(int /*long*/ aWebProgress, int /*long*/ aRequest, int /*long*/ aStatus, int /*long*/ aMessage) {
/*
* Feature in Mozilla. In Mozilla 1.7.5, navigating to an
* HTTPS link without a user profile set causes a crash.
* Most requests for HTTPS pages are aborted in OnStartURIOpen.
* However, https page requests that do not initially specify
* https as their protocol will get past this check since they
* are resolved afterwards. The workaround is to check the url
* whenever there is a status change, and to abort any https
* requests that are detected.
*/
nsIRequest request = new nsIRequest(aRequest);
int /*long*/ aName = XPCOM.nsEmbedCString_new();
request.GetName(aName);
int length = XPCOM.nsEmbedCString_Length(aName);
int /*long*/ buffer = XPCOM.nsEmbedCString_get(aName);
byte[] bytes = new byte[length];
XPCOM.memmove(bytes, buffer, length);
XPCOM.nsEmbedCString_delete(aName);
String value = new String(bytes);
if (value.startsWith(XPCOM.HTTPS_PROTOCOL)) {
request.Cancel(XPCOM.NS_BINDING_ABORTED);
return XPCOM.NS_OK;
}
if (statusTextListeners.length == 0) return XPCOM.NS_OK;
StatusTextEvent event = new StatusTextEvent(this);
event.display = getDisplay();
event.widget = this;
length = XPCOM.strlen_PRUnichar(aMessage);
char[] dest = new char[length];
XPCOM.memmove(dest, aMessage, length * 2);
event.text = new String(dest);
for (int i = 0; i < statusTextListeners.length; i++)
statusTextListeners[i].changed(event);
return XPCOM.NS_OK;
}
int /*long*/ OnSecurityChange(int /*long*/ aWebProgress, int /*long*/ aRequest, int /*long*/ state) {
return XPCOM.NS_OK;
}
/* nsIWebBrowserChrome */
int /*long*/ SetStatus(int /*long*/ statusType, int /*long*/ status) {
StatusTextEvent event = new StatusTextEvent(this);
event.display = getDisplay();
event.widget = this;
int length = XPCOM.strlen_PRUnichar(status);
char[] dest = new char[length];
XPCOM.memmove(dest, status, length * 2);
String string = new String(dest);
if (string == null) string = ""; //$NON-NLS-1$
event.text = string;
for (int i = 0; i < statusTextListeners.length; i++)
statusTextListeners[i].changed(event);
return XPCOM.NS_OK;
}
int /*long*/ GetWebBrowser(int /*long*/ aWebBrowser) {
int /*long*/[] ret = new int /*long*/[1];
if (webBrowser != null) {
webBrowser.AddRef();
ret[0] = webBrowser.getAddress();
}
XPCOM.memmove(aWebBrowser, ret, 4);
return XPCOM.NS_OK;
}
int /*long*/ SetWebBrowser(int /*long*/ aWebBrowser) {
if (webBrowser != null) webBrowser.Release();
webBrowser = aWebBrowser != 0 ? new nsIWebBrowser(aWebBrowser) : null;
return XPCOM.NS_OK;
}
int /*long*/ GetChromeFlags(int /*long*/ aChromeFlags) {
int[] ret = new int[1];
ret[0] = chromeFlags;
/* aChromeFlags is a pointer to a type of size 4 */
XPCOM.memmove(aChromeFlags, ret, 4);
return XPCOM.NS_OK;
}
int /*long*/ SetChromeFlags(int /*long*/ aChromeFlags) {
chromeFlags = (int)/*64*/aChromeFlags;
return XPCOM.NS_OK;
}
int /*long*/ DestroyBrowserWindow() {
WindowEvent newEvent = new WindowEvent(this);
newEvent.display = getDisplay();
newEvent.widget = this;
for (int i = 0; i < closeWindowListeners.length; i++)
closeWindowListeners[i].close(newEvent);
/*
* Note on Mozilla. The DestroyBrowserWindow notification cannot be cancelled.
* The browser widget cannot be used after this notification has been received.
* The application is advised to close the window hosting the browser widget.
* The browser widget must be disposed in all cases.
*/
dispose();
return XPCOM.NS_OK;
}
int /*long*/ SizeBrowserTo(int /*long*/ aCX, int /*long*/ aCY) {
size = new Point((int)/*64*/aCX, (int)/*64*/aCY);
return XPCOM.NS_OK;
}
int /*long*/ ShowAsModal() {
return XPCOM.NS_ERROR_NOT_IMPLEMENTED;
}
int /*long*/ IsWindowModal(int /*long*/ retval) {
// no modal loop
/* Note. boolean remains of size 4 on 64 bit machine */
XPCOM.memmove(retval, new int[] {0}, 4);
return XPCOM.NS_OK;
}
int /*long*/ ExitModalEventLoop(int /*long*/ aStatus) {
return XPCOM.NS_OK;
}
/* nsIEmbeddingSiteWindow */
int /*long*/ SetDimensions(int /*long*/ flags, int /*long*/ x, int /*long*/ y, int /*long*/ cx, int /*long*/ cy) {
if (flags == nsIEmbeddingSiteWindow.DIM_FLAGS_POSITION) location = new Point((int)/*64*/x, (int)/*64*/y);
return XPCOM.NS_OK;
}
int /*long*/ GetDimensions(int /*long*/ flags, int /*long*/ x, int /*long*/ y, int /*long*/ cx, int /*long*/ cy) {
return XPCOM.NS_OK;
}
int /*long*/ SetFocus() {
int /*long*/[] result = new int /*long*/[1];
int rc = webBrowser.QueryInterface(nsIBaseWindow.NS_IBASEWINDOW_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_ERROR_NO_INTERFACE);
nsIBaseWindow baseWindow = new nsIBaseWindow(result[0]);
rc = baseWindow.SetFocus();
if (rc != XPCOM.NS_OK) error(rc);
baseWindow.Release();
/*
* Note. Mozilla notifies here that one of the children took
* focus. This could or should be used to fire an SWT.FOCUS_IN
* event on Browser focus listeners.
*/
return XPCOM.NS_OK;
}
int /*long*/ GetVisibility(int /*long*/ aVisibility) {
return XPCOM.NS_OK;
}
int /*long*/ SetVisibility(int /*long*/ aVisibility) {
WindowEvent event = new WindowEvent(this);
event.display = getDisplay();
event.widget = this;
if (aVisibility == 1) {
/*
* Bug in Mozilla. When the JavaScript window.open is executed, Mozilla
* fires multiple SetVisibility 1 notifications. The workaround is
* to ignore subsequent notifications.
*/
if (!visible) {
visible = true;
event.location = location;
event.size = size;
event.addressBar = addressBar;
event.menuBar = menuBar;
event.statusBar = statusBar;
event.toolBar = toolBar;
for (int i = 0; i < visibilityWindowListeners.length; i++)
visibilityWindowListeners[i].show(event);
location = null;
size = null;
}
} else {
visible = false;
for (int i = 0; i < visibilityWindowListeners.length; i++)
visibilityWindowListeners[i].hide(event);
}
return XPCOM.NS_OK;
}
int /*long*/ GetTitle(int /*long*/ aTitle) {
return XPCOM.NS_OK;
}
int /*long*/ SetTitle(int /*long*/ aTitle) {
if (titleListeners.length == 0) return XPCOM.NS_OK;
TitleEvent event = new TitleEvent(this);
event.display = getDisplay();
event.widget = this;
int length = XPCOM.strlen_PRUnichar(aTitle);
char[] dest = new char[length];
XPCOM.memmove(dest, aTitle, length * 2);
event.title = new String(dest);
for (int i = 0; i < titleListeners.length; i++)
titleListeners[i].changed(event);
return XPCOM.NS_OK;
}
int /*long*/ GetSiteWindow(int /*long*/ aSiteWindow) {
/*
* Note. The handle is expected to be an HWND on Windows and
* a GtkWidget* on GTK. This callback is invoked on Windows
* when the javascript window.print is invoked and the print
* dialog comes up. If no handle is returned, the print dialog
* does not come up on this platform.
*/
XPCOM.memmove(aSiteWindow, new int /*long*/[] {gtkHandle}, 4);
return XPCOM.NS_OK;
}
/* nsIWebBrowserChromeFocus */
int /*long*/ FocusNextElement() {
/*
* Bug in Mozilla embedding API. Mozilla takes back the focus after sending
* this event. This prevents tabbing out of Mozilla. This behaviour can be reproduced
* with the Mozilla application TestGtkEmbed. The workaround is to
* send the traversal notification after this callback returns.
*/
getDisplay().asyncExec(new Runnable() {
public void run() {
traverse(SWT.TRAVERSE_TAB_NEXT);
}
});
return XPCOM.NS_OK;
}
int /*long*/ FocusPrevElement() {
/*
* Bug in Mozilla embedding API. Mozilla takes back the focus after sending
* this event. This prevents tabbing out of Mozilla. This behaviour can be reproduced
* with the Mozilla application TestGtkEmbed. The workaround is to
* send the traversal notification after this callback returns.
*/
getDisplay().asyncExec(new Runnable() {
public void run() {
traverse(SWT.TRAVERSE_TAB_PREVIOUS);
}
});
return XPCOM.NS_OK;
}
/* nsIContextMenuListener */
int /*long*/ OnShowContextMenu(int /*long*/ aContextFlags, int /*long*/ aEvent, int /*long*/ aNode) {
nsIDOMEvent domEvent = new nsIDOMEvent(aEvent);
int /*long*/[] result = new int /*long*/[1];
int rc = domEvent.QueryInterface(nsIDOMMouseEvent.NS_IDOMMOUSEEVENT_IID, result);
if (rc != XPCOM.NS_OK) error(rc);
if (result[0] == 0) error(XPCOM.NS_NOINTERFACE);
nsIDOMMouseEvent domMouseEvent = new nsIDOMMouseEvent(result[0]);
int[] aScreenX = new int[1], aScreenY = new int[1];
rc = domMouseEvent.GetScreenX(aScreenX);
if (rc != XPCOM.NS_OK) error(rc);
rc = domMouseEvent.GetScreenY(aScreenY);
if (rc != XPCOM.NS_OK) error(rc);
domMouseEvent.Release();
Event event = new Event();
event.x = aScreenX[0];
event.y = aScreenY[0];
notifyListeners(SWT.MenuDetect, event);
if (!event.doit) return XPCOM.NS_OK;
Menu menu = getMenu();
if (menu != null && !menu.isDisposed ()) {
if (aScreenX[0] != event.x || aScreenY[0] != event.y) {
menu.setLocation (event.x, event.y);
}
menu.setVisible (true);
}
return XPCOM.NS_OK;
}
/* nsIURIContentListener */
int /*long*/ OnStartURIOpen(int /*long*/ aURI, int /*long*/ retval) {
nsIURI location = new nsIURI(aURI);
int /*long*/ aSpec = XPCOM.nsEmbedCString_new();
location.GetSpec(aSpec);
int length = XPCOM.nsEmbedCString_Length(aSpec);
int /*long*/ buffer = XPCOM.nsEmbedCString_get(aSpec);
buffer = XPCOM.nsEmbedCString_get(aSpec);
byte[] dest = new byte[length];
XPCOM.memmove(dest, buffer, length);
XPCOM.nsEmbedCString_delete(aSpec);
String value = new String(dest);
/*
* Feature in Mozilla. In Mozilla 1.7.5, navigating to an
* HTTPS link without a user profile set causes a crash.
* HTTPS requires a user profile to be set to persist security
* information. This requires creating a new user profile
* (i.e. creating a new folder) or locking an existing Mozilla
* user profile. The Mozilla Profile API is not frozen and it is not
* currently implemented. The workaround is to not load
* HTTPS resources to avoid the crash.
*/
boolean isHttps = value.startsWith(XPCOM.HTTPS_PROTOCOL);
if (locationListeners.length == 0) {
XPCOM.memmove(retval, new int[] {isHttps ? 1 : 0}, 4);
return XPCOM.NS_OK;
}
boolean doit = !isHttps;
if (request == 0) {
LocationEvent event = new LocationEvent(this);
event.display = getDisplay();
event.widget = this;
event.location = value;
event.doit = doit;
for (int i = 0; i < locationListeners.length; i++)
locationListeners[i].changing(event);
if (!isHttps) doit = event.doit;
}
/* Note. boolean remains of size 4 on 64 bit machine */
XPCOM.memmove(retval, new int[] {doit ? 0 : 1}, 4);
return XPCOM.NS_OK;
}
int /*long*/ DoContent(int /*long*/ aContentType, int /*long*/ aIsContentPreferred, int /*long*/ aRequest, int /*long*/ aContentHandler, int /*long*/ retval) {
return XPCOM.NS_ERROR_NOT_IMPLEMENTED;
}
int /*long*/ IsPreferred(int /*long*/ aContentType, int /*long*/ aDesiredContentType, int /*long*/ retval) {
/* Note. boolean remains of size 4 on 64 bit machine */
XPCOM.memmove(retval, new int[] {1}, 4);
return XPCOM.NS_OK;
}
int /*long*/ CanHandleContent(int /*long*/ aContentType, int /*long*/ aIsContentPreferred, int /*long*/ aDesiredContentType, int /*long*/ retval) {
return XPCOM.NS_ERROR_NOT_IMPLEMENTED;
}
int /*long*/ GetLoadCookie(int /*long*/ aLoadCookie) {
return XPCOM.NS_ERROR_NOT_IMPLEMENTED;
}
int /*long*/ SetLoadCookie(int /*long*/ aLoadCookie) {
return XPCOM.NS_ERROR_NOT_IMPLEMENTED;
}
int /*long*/ GetParentContentListener(int /*long*/ aParentContentListener) {
return XPCOM.NS_ERROR_NOT_IMPLEMENTED;
}
int /*long*/ SetParentContentListener(int /*long*/ aParentContentListener) {
return XPCOM.NS_ERROR_NOT_IMPLEMENTED;
}
/* nsITooltipListener */
int /*long*/ OnShowTooltip(int /*long*/ aXCoords, int /*long*/ aYCoords, int /*long*/ aTipText) {
int length = XPCOM.strlen_PRUnichar(aTipText);
char[] dest = new char[length];
XPCOM.memmove(dest, aTipText, length * 2);
String text = new String(dest);
if (tip != null && !tip.isDisposed()) tip.dispose();
Display display = getDisplay();
Shell parent = getShell();
tip = new Shell(parent, SWT.ON_TOP);
tip.setLayout(new FillLayout());
Label label = new Label(tip, SWT.CENTER);
label.setForeground(display.getSystemColor(SWT.COLOR_INFO_FOREGROUND));
label.setBackground(display.getSystemColor(SWT.COLOR_INFO_BACKGROUND));
label.setText(text);
/*
* Bug in Mozilla embedded API. Tooltip coordinates are wrong for
* elements inside an inline frame (IFrame tag). The workaround is
* to position the tooltip based on the mouse cursor location.
*/
Point point = display.getCursorLocation();
/* Assuming cursor is 21x21 because this is the size of
* the arrow cursor on Windows
*/
point.y += 21;
tip.setLocation(point);
tip.pack();
tip.setVisible(true);
return XPCOM.NS_OK;
}
int /*long*/ OnHideTooltip() {
if (tip != null && !tip.isDisposed()) tip.dispose();
tip = null;
return XPCOM.NS_OK;
}
}