blob: 64971d17d67c579eeb6547f70202600f848b3724 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2017 IBM Corporation and others.
*
* This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.swt.ole.win32;
import org.eclipse.swt.*;
import org.eclipse.swt.internal.*;
import org.eclipse.swt.internal.ole.win32.*;
import org.eclipse.swt.internal.win32.*;
final class OleEventSink
{
private OleControlSite widget;
private COMObject iDispatch;
private int refCount;
private IUnknown objIUnknown;
private int eventCookie;
private GUID eventGuid;
private OleEventTable eventTable;
OleEventSink(OleControlSite widget, long iUnknown, GUID riid) {
this.widget = widget;
this.eventGuid = riid;
this.objIUnknown = new IUnknown(iUnknown);
createCOMInterfaces();
}
void connect () {
long[] ppvObject = new long[1];
if (objIUnknown.QueryInterface(COM.IIDIConnectionPointContainer, ppvObject) == COM.S_OK) {
IConnectionPointContainer cpc = new IConnectionPointContainer(ppvObject[0]);
long[] ppCP = new long[1];
if (cpc.FindConnectionPoint(eventGuid, ppCP) == COM.S_OK) {
IConnectionPoint cp = new IConnectionPoint(ppCP[0]);
int[] pCookie = new int[1];
if (cp.Advise(iDispatch.getAddress(), pCookie) == COM.S_OK)
eventCookie = pCookie[0];
cp.Release();
}
cpc.Release();
}
}
void addListener(int eventID, OleListener listener) {
if (listener == null) OLE.error (SWT.ERROR_NULL_ARGUMENT);
if (eventTable == null) eventTable = new OleEventTable ();
eventTable.hook(eventID, listener);
}
int AddRef() {
refCount++;
return refCount;
}
private void createCOMInterfaces() {
iDispatch = new COMObject(new int[]{2, 0, 0, 1, 3, 4, 8}){
@Override
public long method0(long[] args) {return QueryInterface(args[0], args[1]);}
@Override
public long method1(long[] args) {return AddRef();}
@Override
public long method2(long[] args) {return Release();}
// method3 GetTypeInfoCount - not implemented
// method4 GetTypeInfo - not implemented
// method5 GetIDsOfNames - not implemented
@Override
public long method6(long[] args) {return Invoke((int)args[0], args[1], (int)args[2], (int)args[3], args[4], args[5], args[6], args[7]);}
};
}
void disconnect() {
// disconnect event sink
if (eventCookie != 0 && objIUnknown != null) {
long[] ppvObject = new long[1];
if (objIUnknown.QueryInterface(COM.IIDIConnectionPointContainer, ppvObject) == COM.S_OK) {
IConnectionPointContainer cpc = new IConnectionPointContainer(ppvObject[0]);
if (cpc.FindConnectionPoint(eventGuid, ppvObject) == COM.S_OK) {
IConnectionPoint cp = new IConnectionPoint(ppvObject[0]);
if (cp.Unadvise(eventCookie) == COM.S_OK) {
eventCookie = 0;
}
cp.Release();
}
cpc.Release();
}
}
}
private void disposeCOMInterfaces() {
if (iDispatch != null)
iDispatch.dispose();
iDispatch = null;
}
private int Invoke(int dispIdMember, long riid, int lcid, int dwFlags, long pDispParams, long pVarResult, long pExcepInfo, long pArgErr)
{
if (eventTable == null || !eventTable.hooks(dispIdMember)) return COM.S_OK;
// Construct an array of the parameters that are passed in
// Note: parameters are passed in reverse order - here we will correct the order
Variant[] eventInfo = null;
if (pDispParams != 0) {
DISPPARAMS dispParams = new DISPPARAMS();
COM.MoveMemory(dispParams, pDispParams, DISPPARAMS.sizeof);
eventInfo = new Variant[dispParams.cArgs];
int size = VARIANT.sizeof;
long offset = (dispParams.cArgs - 1) * size;
for (int j = 0; j < dispParams.cArgs; j++){
eventInfo[j] = new Variant();
eventInfo[j].setData(dispParams.rgvarg + offset);
offset = offset - size;
}
}
OleEvent event = new OleEvent();
event.arguments = eventInfo;
notifyListener(dispIdMember,event);
if (eventInfo != null) {
for (Variant element : eventInfo) {
element.dispose();
}
}
return COM.S_OK;
}
/**
* Notify listeners of an event.
* <p>
* This method notifies all listeners that an event
* has occurred.
*
* @param eventType the desired SWT event
* @param event the event data
*
* @exception IllegalArgumentException <ul>
* <li>ERROR_NULL_ARGUMENT when handler 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>
*/
private void notifyListener (int eventType, OleEvent event) {
if (event == null) OLE.error (SWT.ERROR_NULL_ARGUMENT);
if (eventTable == null) return;
event.type = eventType;
event.widget = widget;
eventTable.sendEvent (event);
}
private int QueryInterface(long riid, long ppvObject) {
if (riid == 0 || ppvObject == 0)
return COM.E_INVALIDARG;
GUID guid = new GUID();
COM.MoveMemory(guid, riid, GUID.sizeof);
if ( COM.IsEqualGUID(guid, COM.IIDIUnknown) || COM.IsEqualGUID(guid, COM.IIDIDispatch) ||
COM.IsEqualGUID(guid, eventGuid)) {
OS.MoveMemory(ppvObject, new long[] {iDispatch.getAddress()}, C.PTR_SIZEOF);
AddRef();
return OLE.S_OK;
}
OS.MoveMemory(ppvObject, new long[] {0}, C.PTR_SIZEOF);
return COM.E_NOINTERFACE;
}
int Release() {
refCount--;
if (refCount == 0) {
disposeCOMInterfaces();
}
return refCount;
}
void removeListener(int eventID, OleListener listener) {
if (listener == null) OLE.error (SWT.ERROR_NULL_ARGUMENT);
if (eventTable == null) return;
eventTable.unhook (eventID, listener);
}
boolean hasListeners() {
return eventTable.hasEntries();
}
}