]> gerrit.simantics Code Review - simantics/platform.git/blobdiff - bundles/org.eclipse.swt.win32.win32.x86_64/src/org/eclipse/swt/ole/win32/OleFrame.java
Work around SWT 4.13 - 4.18 Win32 DnD bug 567422
[simantics/platform.git] / bundles / org.eclipse.swt.win32.win32.x86_64 / src / org / eclipse / swt / ole / win32 / OleFrame.java
diff --git a/bundles/org.eclipse.swt.win32.win32.x86_64/src/org/eclipse/swt/ole/win32/OleFrame.java b/bundles/org.eclipse.swt.win32.win32.x86_64/src/org/eclipse/swt/ole/win32/OleFrame.java
new file mode 100644 (file)
index 0000000..b6bd46c
--- /dev/null
@@ -0,0 +1,823 @@
+/*******************************************************************************
+ * Copyright (c) 2000, 2019 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 java.util.*;
+import java.util.List;
+
+import org.eclipse.swt.*;
+import org.eclipse.swt.internal.*;
+import org.eclipse.swt.internal.ole.win32.*;
+import org.eclipse.swt.internal.win32.*;
+import org.eclipse.swt.widgets.*;
+
+/**
+ *
+ * OleFrame is an OLE Container's top level frame.
+ *
+ * <p>This object implements the OLE Interfaces IUnknown and IOleInPlaceFrame
+ *
+ * <p>OleFrame allows the container to do the following: <ul>
+ *     <li>position and size the ActiveX Control or OLE Document within the application
+ *     <li>insert menu items from the application into the OLE Document's menu
+ *     <li>activate and deactivate the OLE Document's menus
+ *     <li>position the OLE Document's menu in the application
+ *     <li>translate accelerator keystrokes intended for the container's frame</ul>
+ *
+ * <dl>
+ *     <dt><b>Styles</b> <dd>BORDER
+ *     <dt><b>Events</b> <dd>Dispose, Move, Resize
+ * </dl>
+ *
+ * @see <a href="http://www.eclipse.org/swt/snippets/#ole">OLE and ActiveX snippets</a>
+ * @see <a href="http://www.eclipse.org/swt/examples.php">SWT Examples: OLEExample, OleWebBrowser</a>
+ */
+final public class OleFrame extends Composite
+{
+       // Interfaces for this Ole Client Container
+       private COMObject iOleInPlaceFrame;
+
+       // Access to the embedded/linked Ole Object
+       private IOleInPlaceActiveObject objIOleInPlaceActiveObject;
+
+       private OleClientSite currentdoc;
+
+       private int refCount = 0;
+
+       private MenuItem[] fileMenuItems;
+       private MenuItem[] containerMenuItems;
+       private MenuItem[] windowMenuItems;
+
+       private Listener listener;
+
+       private long shellHandle;
+       private long oldMenuHandle;
+       private long newMenuHandle;
+       private static long lastActivatedMenuHandle;
+
+       private static String CHECK_FOCUS = "OLE_CHECK_FOCUS"; //$NON-NLS-1$
+       private static String HHOOK = "OLE_HHOOK"; //$NON-NLS-1$
+       private static String HHOOKMSG = "OLE_HHOOK_MSG"; //$NON-NLS-1$
+
+       private static boolean ignoreNextKey;
+       private static final short [] ACCENTS = new short [] {'~', '`', '\'', '^', '"'};
+
+       private static final String CONSUME_KEY = "org.eclipse.swt.OleFrame.ConsumeKey"; //$NON-NLS-1$
+       private static final String ACCEL_KEY_HIT = "org.eclipse.swt.internal.win32.accelKeyHit"; //$NON-NLS-1$
+
+/**
+ * Create an OleFrame child widget using style bits
+ * to select a particular look or set of properties.
+ *
+ * @param parent a composite widget (cannot be null)
+ * @param style the bitwise OR'ing of widget styles
+ *
+ * @exception IllegalArgumentException <ul>
+ *     <li>ERROR_NULL_ARGUMENT when the parent is null
+ * </ul>
+ * @exception SWTException <ul>
+ *     <li>ERROR_THREAD_INVALID_ACCESS when called from the wrong thread
+ * </ul>
+ *
+ */
+public OleFrame(Composite parent, int style) {
+       super(parent, style);
+
+       createCOMInterfaces();
+
+       // setup cleanup proc
+       listener = e -> {
+               switch (e.type) {
+               case SWT.Activate :    onActivate(e); break;
+               case SWT.Deactivate :  onDeactivate(e); break;
+               case SWT.Dispose :  onDispose(e); break;
+               case SWT.Resize :
+               case SWT.Move :     onResize(e); break;
+               default :
+                       OLE.error(SWT.ERROR_NOT_IMPLEMENTED);
+               }
+       };
+
+
+       addListener(SWT.Activate, listener);
+       addListener(SWT.Deactivate, listener);
+       addListener(SWT.Dispose, listener);
+
+       // inform inplaceactiveobject whenever frame resizes
+       addListener(SWT.Resize, listener);
+
+       // inform inplaceactiveobject whenever frame moves
+       addListener(SWT.Move, listener);
+
+       // Maintain a reference to yourself so that when
+       // ClientSites close, they don't take the frame away
+       // with them.
+       this.AddRef();
+
+       // Check for focus change
+       Display display = getDisplay();
+       initCheckFocus(display);
+       initMsgHook(display);
+}
+private static void initCheckFocus (final Display display) {
+       if (display.getData(CHECK_FOCUS) != null) return;
+       display.setData(CHECK_FOCUS, CHECK_FOCUS);
+       final int time = 50;
+       final Runnable[] timer = new Runnable[1];
+       final Control[] lastFocus = new Control[1];
+       timer[0] = () -> {
+               if (lastFocus[0] instanceof OleClientSite && !lastFocus[0].isDisposed()) {
+                       // ignore popup menus and dialogs
+                       long hwnd = OS.GetFocus();
+                       while (hwnd != 0) {
+                               long ownerHwnd = OS.GetWindow(hwnd, OS.GW_OWNER);
+                               if (ownerHwnd != 0) {
+                                       display.timerExec(time, timer[0]);
+                                       return;
+                               }
+                               hwnd = OS.GetParent(hwnd);
+                       }
+               }
+               if (lastFocus[0] == null || lastFocus[0].isDisposed() || !lastFocus[0].isFocusControl()) {
+                       Control currentFocus = display.getFocusControl();
+                       if (currentFocus instanceof OleFrame) {
+                               OleFrame frame = (OleFrame) currentFocus;
+                               currentFocus = frame.getCurrentDocument();
+                       }
+                       if (lastFocus[0] != currentFocus) {
+                               Event event = new Event();
+                               if (lastFocus[0] instanceof OleClientSite && !lastFocus[0].isDisposed()) {
+                                       lastFocus[0].notifyListeners (SWT.FocusOut, event);
+                               }
+                               if (currentFocus instanceof OleClientSite && !currentFocus.isDisposed()) {
+                                       currentFocus.notifyListeners(SWT.FocusIn, event);
+                               }
+                       }
+                       lastFocus[0] = currentFocus;
+               }
+               display.timerExec(time, timer[0]);
+       };
+       display.timerExec(time, timer[0]);
+}
+private static void initMsgHook(Display display) {
+       if (display.getData(HHOOK) != null) return;
+       final Callback callback = new Callback(OleFrame.class, "getMsgProc", 3); //$NON-NLS-1$
+       long address = callback.getAddress();
+       if (address == 0) SWT.error(SWT.ERROR_NO_MORE_CALLBACKS);
+       int threadId = OS.GetCurrentThreadId();
+       final long hHook = OS.SetWindowsHookEx(OS.WH_GETMESSAGE, address, 0, threadId);
+       if (hHook == 0) {
+               callback.dispose();
+               return;
+       }
+       display.setData(HHOOK, new LONG(hHook));
+       display.setData(HHOOKMSG, new MSG());
+       display.disposeExec(() -> {
+               if (hHook != 0) OS.UnhookWindowsHookEx(hHook);
+               if (callback != null) callback.dispose();
+       });
+}
+static long getMsgProc(long code, long wParam, long lParam) {
+       Display display = Display.getCurrent();
+       if (display == null) return 0;
+       LONG hHook = (LONG)display.getData(HHOOK);
+       if (hHook == null) return 0;
+       if (code < 0 || (wParam & OS.PM_REMOVE) == 0) {
+               return OS.CallNextHookEx(hHook.value, (int)code, wParam, lParam);
+       }
+       MSG msg = (MSG)display.getData(HHOOKMSG);
+       OS.MoveMemory(msg, lParam, MSG.sizeof);
+       int message = msg.message;
+       if (OS.WM_KEYFIRST <= message && message <= OS.WM_KEYLAST) {
+               if (display != null) {
+                       Widget widget = null;
+                       long hwnd = msg.hwnd;
+                       while (hwnd != 0) {
+                               widget = display.findWidget (hwnd);
+                               if (widget != null) break;
+                               hwnd = OS.GetParent (hwnd);
+                       }
+                       if (widget != null && widget instanceof OleClientSite) {
+                               OleClientSite site = (OleClientSite)widget;
+                               if (site.handle == hwnd) {
+                                       boolean consumed = false;
+                                       /* Allow activeX control to translate accelerators except when a menu is active. */
+                                       int thread = OS.GetWindowThreadProcessId(msg.hwnd, null);
+                                       GUITHREADINFO  lpgui = new GUITHREADINFO();
+                                       lpgui.cbSize = GUITHREADINFO.sizeof;
+                                       boolean rc = OS.GetGUIThreadInfo(thread, lpgui);
+                                       int mask = OS.GUI_INMENUMODE | OS.GUI_INMOVESIZE | OS.GUI_POPUPMENUMODE | OS.GUI_SYSTEMMENUMODE;
+                                       if (!rc || (lpgui.flags & mask) == 0) {
+                                               OleFrame frame = site.frame;
+                                               frame.setData(CONSUME_KEY, null);
+                                               display.setData(ACCEL_KEY_HIT, Boolean.TRUE);
+                                               consumed = frame.translateOleAccelerator(msg);
+                                               /* translateOleAccelerator() may send client events, ensure that the frame and display are still valid */
+                                               if (display.isDisposed()) return 0;
+                                               display.setData(ACCEL_KEY_HIT, Boolean.FALSE);
+                                               if (frame.isDisposed()) return 0;
+                                               String value = (String)frame.getData(CONSUME_KEY);
+                                               if (value != null) consumed = value.equals("true"); //$NON-NLS-1$
+                                               frame.setData(CONSUME_KEY, null);
+                                       }
+                                       boolean accentKey = false;
+                                       switch (msg.message) {
+                                               case OS.WM_KEYDOWN:
+                                               case OS.WM_SYSKEYDOWN: {
+                                                       switch ((int)msg.wParam) {
+                                                               case OS.VK_SHIFT:
+                                                               case OS.VK_MENU:
+                                                               case OS.VK_CONTROL:
+                                                               case OS.VK_CAPITAL:
+                                                               case OS.VK_NUMLOCK:
+                                                               case OS.VK_SCROLL:
+                                                                       break;
+                                                               default: {
+                                                                       int mapKey = OS.MapVirtualKey ((int)msg.wParam, 2);
+                                                                       if (mapKey != 0) {
+                                                                               accentKey = (mapKey & 0x80000000) != 0;
+                                                                               if (!accentKey) {
+                                                                                       for (int i=0; i<ACCENTS.length; i++) {
+                                                                                               int value = OS.VkKeyScan (ACCENTS [i]);
+                                                                                               if (value != -1 && (value & 0xFF) == msg.wParam) {
+                                                                                                       int state = value >> 8;
+                                                                                                       if ((OS.GetKeyState (OS.VK_SHIFT) < 0) == ((state & 0x1) != 0) &&
+                                                                                                               (OS.GetKeyState (OS.VK_CONTROL) < 0) == ((state & 0x2) != 0) &&
+                                                                                                               (OS.GetKeyState (OS.VK_MENU) < 0) == ((state & 0x4) != 0)) {
+                                                                                                                       if ((state & 0x7) != 0) accentKey = true;
+                                                                                                                       break;
+                                                                                                       }
+                                                                                               }
+                                                                                       }
+                                                                               }
+                                                                       }
+                                                                       break;
+                                                               }
+                                                       }
+                                                       break;
+                                               }
+                                       }
+                                       /* Allow OleClientSite to process key events before activeX control */
+                                       if (!consumed && !accentKey && !ignoreNextKey) {
+                                               long hwndOld = msg.hwnd;
+                                               msg.hwnd = site.handle;
+                                               consumed = OS.DispatchMessage (msg) == 1;
+                                               msg.hwnd = hwndOld;
+                                       }
+                                       switch (msg.message) {
+                                               case OS.WM_KEYDOWN:
+                                               case OS.WM_SYSKEYDOWN: {
+                                                       switch ((int)msg.wParam) {
+                                                               case OS.VK_SHIFT:
+                                                               case OS.VK_MENU:
+                                                               case OS.VK_CONTROL:
+                                                               case OS.VK_CAPITAL:
+                                                               case OS.VK_NUMLOCK:
+                                                               case OS.VK_SCROLL:
+                                                                       break;
+                                                               default: {
+                                                                       ignoreNextKey = accentKey;
+                                                                       break;
+                                                               }
+                                                       }
+                                               }
+                                       }
+
+                                       if (consumed) {
+                                               // In order to prevent this message from also being processed
+                                               // by the application, zero out message, wParam and lParam
+                                               msg.message = OS.WM_NULL;
+                                               msg.wParam = msg.lParam = 0;
+                                               OS.MoveMemory(lParam, msg, MSG.sizeof);
+                                               return 0;
+                                       }
+                               }
+                       }
+               }
+       }
+       return OS.CallNextHookEx(hHook.value, (int)code, wParam, lParam);
+}
+/**
+ * Increment the count of references to this instance
+ *
+ * @return the current reference count
+ */
+int AddRef() {
+       refCount++;
+       return refCount;
+}
+private int ContextSensitiveHelp(int fEnterMode) {
+       return COM.S_OK;
+}
+private void createCOMInterfaces() {
+       iOleInPlaceFrame = new COMObject(new int[]{2, 0, 0, 1, 1, 1, 1, 1, 2, 2, 3, 1, 1, 1, 2}){
+               @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();}
+               @Override
+               public long method3(long[] args) {return GetWindow(args[0]);}
+               @Override
+               public long method4(long[] args) {return ContextSensitiveHelp((int)args[0]);}
+               @Override
+               public long method5(long[] args) {return GetBorder(args[0]);}
+               @Override
+               public long method6(long[] args) {return RequestBorderSpace(args[0]);}
+               @Override
+               public long method7(long[] args) {return SetBorderSpace(args[0]);}
+               @Override
+               public long method8(long[] args) {return SetActiveObject(args[0], args[1]);}
+               @Override
+               public long method9(long[] args) {return InsertMenus(args[0], args[1]);}
+               @Override
+               public long method10(long[] args) {return SetMenu(args[0], args[1], args[2]);}
+               @Override
+               public long method11(long[] args) {return RemoveMenus(args[0]);}
+               // method12 SetStatusText - not implemented
+               // method13 EnableModeless - not implemented
+               @Override
+               public long method14(long[] args) {return TranslateAccelerator(args[0], (int)args[1]);}
+       };
+}
+private void disposeCOMInterfaces () {
+       if (iOleInPlaceFrame != null)
+               iOleInPlaceFrame.dispose();
+       iOleInPlaceFrame = null;
+}
+private int GetBorder(long lprectBorder) {
+       /*
+       The IOleInPlaceUIWindow::GetBorder function, when called on a document or frame window
+       object, returns the outer rectangle (relative to the window) where the object can put
+       toolbars or similar controls.
+       */
+       if (lprectBorder == 0) return COM.E_INVALIDARG;
+       RECT rectBorder = new RECT();
+       // Coordinates must be relative to the window
+       OS.GetClientRect(handle, rectBorder);
+       OS.MoveMemory(lprectBorder, rectBorder, RECT.sizeof);
+       return COM.S_OK;
+}
+/**
+ *
+ * Returns the application menu items that will appear in the Container location when an OLE Document
+ * is in-place activated.
+ *
+ * <p>When an OLE Document is in-place active, the Document provides its own menus but the application
+ * is given the opportunity to merge some of its menus into the menubar.  The application
+ * is allowed to insert its menus in three locations: File (far left), Container(middle) and Window
+ * (far right just before Help).  The OLE Document retains control of the Edit, Object and Help
+ * menu locations.  Note that an application can insert more than one menu into a single location.
+ *
+ * @return the application menu items that will appear in the Container location when an OLE Document
+ *         is in-place activated.
+ *
+ */
+public MenuItem[] getContainerMenus(){
+       return containerMenuItems;
+}
+/**
+ *
+ * Returns the application menu items that will appear in the File location when an OLE Document
+ * is in-place activated.
+ *
+ * <p>When an OLE Document is in-place active, the Document provides its own menus but the application
+ * is given the opportunity to merge some of its menus into the menubar.  The application
+ * is allowed to insert its menus in three locations: File (far left), Container(middle) and Window
+ * (far right just before Help).  The OLE Document retains control of the Edit, Object and Help
+ * menu locations.  Note that an application can insert more than one menu into a single location.
+ *
+ * @return the application menu items that will appear in the File location when an OLE Document
+ *         is in-place activated.
+ *
+ */
+public MenuItem[] getFileMenus(){
+       return fileMenuItems;
+}
+long getIOleInPlaceFrame() {
+       return iOleInPlaceFrame.getAddress();
+}
+private long getMenuItemID(long hMenu, int index) {
+       long id = 0;
+       MENUITEMINFO lpmii = new MENUITEMINFO();
+       lpmii.cbSize = MENUITEMINFO.sizeof;
+       lpmii.fMask = OS.MIIM_STATE | OS.MIIM_SUBMENU | OS.MIIM_ID;
+       OS.GetMenuItemInfo(hMenu, index, true, lpmii);
+       if ((lpmii.fState & OS.MF_POPUP) == OS.MF_POPUP) {
+               id = lpmii.hSubMenu;
+       } else {
+               id = lpmii.wID;
+       }
+       return id;
+}
+private int GetWindow(long phwnd) {
+       if (phwnd != 0) {
+               OS.MoveMemory(phwnd, new long[] {handle}, C.PTR_SIZEOF);
+       }
+       return COM.S_OK;
+}
+/**
+ *
+ * Returns the application menu items that will appear in the Window location when an OLE Document
+ * is in-place activated.
+ *
+ * <p>When an OLE Document is in-place active, the Document provides its own menus but the application
+ * is given the opportunity to merge some of its menus into the menubar.  The application
+ * is allowed to insert its menus in three locations: File (far left), Container(middle) and Window
+ * (far right just before Help).  The OLE Document retains control of the Edit, Object and Help
+ * menu locations.  Note that an application can insert more than one menu into a single location.
+ *
+ * @return the application menu items that will appear in the Window location when an OLE Document
+ *         is in-place activated.
+ *
+ */
+public MenuItem[] getWindowMenus(){
+       return windowMenuItems;
+}
+private int InsertMenus(long hmenuShared, long lpMenuWidths) {
+       // locate menu bar
+       Menu menubar = getShell().getMenuBar();
+       if (menubar == null || menubar.isDisposed()) {
+               OS.MoveMemory(lpMenuWidths, new int[] {0}, 4);
+               return COM.S_OK;
+       }
+       long hMenu = menubar.handle;
+
+       // Create a holder for menu information.  This will be passed down to
+       // the OS and the OS will fill in the requested information for each menu.
+       MENUITEMINFO lpmii = new MENUITEMINFO();
+       long hHeap = OS.GetProcessHeap();
+       int cch = 128;
+       int byteCount = cch * TCHAR.sizeof;
+       long pszText = OS.HeapAlloc(hHeap, OS.HEAP_ZERO_MEMORY, byteCount);
+       lpmii.cbSize = MENUITEMINFO.sizeof;
+       lpmii.fMask = OS.MIIM_STATE | OS.MIIM_ID | OS.MIIM_TYPE | OS.MIIM_SUBMENU | OS.MIIM_DATA;
+       lpmii.dwTypeData = pszText;
+       lpmii.cch = cch;
+
+       // Loop over all "File-like" menus in the menubar and get information about the
+       // item from the OS.
+       int fileMenuCount = 0;
+       int newindex = 0;
+       if (this.fileMenuItems != null) {
+               for (int i = 0; i < this.fileMenuItems.length; i++) {
+                       MenuItem item = this.fileMenuItems[i];
+                       if (item != null) {
+                               int index = item.getParent().indexOf(item);
+                               lpmii.cch = cch;  // lpmii.cch gets updated by GetMenuItemInfo to indicate the
+                                                 // exact number of characters in name.  Reset it to our max size
+                                                 // before each call.
+                               if (OS.GetMenuItemInfo(hMenu, index, true, lpmii)) {
+                                       if (OS.InsertMenuItem(hmenuShared, newindex, true, lpmii)) {
+                                               // keep track of the number of items
+                                               fileMenuCount++;
+                                               newindex++;
+                                       }
+                               }
+                       }
+               }
+       }
+
+       // copy the menu item count information to the pointer
+       OS.MoveMemory(lpMenuWidths, new int[] {fileMenuCount}, 4);
+
+       // Loop over all "Container-like" menus in the menubar and get information about the
+       // item from the OS.
+       int containerMenuCount = 0;
+       if (this.containerMenuItems != null) {
+               for (int i = 0; i < this.containerMenuItems.length; i++) {
+                       MenuItem item = this.containerMenuItems[i];
+                       if (item != null) {
+                               int index = item.getParent().indexOf(item);
+                               lpmii.cch = cch; // lpmii.cch gets updated by GetMenuItemInfo to indicate the
+                                                          // exact number of characters in name.  Reset it to a large number
+                                                          // before each call.
+                               if (OS.GetMenuItemInfo(hMenu, index, true, lpmii)) {
+                                       if (OS.InsertMenuItem(hmenuShared, newindex, true, lpmii)) {
+                                               // keep track of the number of items
+                                               containerMenuCount++;
+                                               newindex++;
+                                       }
+                               }
+                       }
+               }
+       }
+
+       // copy the menu item count information to the pointer
+       OS.MoveMemory(lpMenuWidths + 8, new int[] {containerMenuCount}, 4);
+
+       // Loop over all "Window-like" menus in the menubar and get information about the
+       // item from the OS.
+       int windowMenuCount = 0;
+       if (this.windowMenuItems != null) {
+               for (int i = 0; i < this.windowMenuItems.length; i++) {
+                       MenuItem item = this.windowMenuItems[i];
+                       if (item != null) {
+                               int index = item.getParent().indexOf(item);
+                               lpmii.cch = cch; // lpmii.cch gets updated by GetMenuItemInfo to indicate the
+                                                          // exact number of characters in name.  Reset it to a large number
+                                                          // before each call.
+                               if (OS.GetMenuItemInfo(hMenu, index, true, lpmii)) {
+                                       if (OS.InsertMenuItem(hmenuShared, newindex, true, lpmii)) {
+                                               // keep track of the number of items
+                                               windowMenuCount++;
+                                               newindex++;
+                                       }
+                               }
+                       }
+               }
+       }
+
+       // copy the menu item count information to the pointer
+       OS.MoveMemory(lpMenuWidths + 16, new int[] {windowMenuCount}, 4);
+
+       // free resources used in querying the OS
+       if (pszText != 0)
+               OS.HeapFree(hHeap, 0, pszText);
+       return COM.S_OK;
+}
+void onActivate(Event e) {
+       if (objIOleInPlaceActiveObject != null) {
+               objIOleInPlaceActiveObject.OnFrameWindowActivate(true);
+       }
+}
+void onDeactivate(Event e) {
+       if (objIOleInPlaceActiveObject != null) {
+               objIOleInPlaceActiveObject.OnFrameWindowActivate(false);
+       }
+}
+private void onDispose(Event e) {
+
+       releaseObjectInterfaces();
+       currentdoc = null;
+
+       this.Release();
+       removeListener(SWT.Activate, listener);
+       removeListener(SWT.Deactivate, listener);
+       removeListener(SWT.Dispose, listener);
+       removeListener(SWT.Resize, listener);
+       removeListener(SWT.Move, listener);
+}
+void onFocusIn(Event e) {
+       if (lastActivatedMenuHandle != newMenuHandle)
+               currentdoc.doVerb(OLE.OLEIVERB_SHOW);
+       if (OS.GetMenu(shellHandle) != newMenuHandle)
+               OS.SetMenu(shellHandle, newMenuHandle);
+}
+void onFocusOut(Event e) {
+       Control control = getDisplay().getFocusControl();
+       if (OS.GetMenu(shellHandle) != oldMenuHandle && control != null && control.handle != shellHandle)
+               OS.SetMenu(shellHandle, oldMenuHandle);
+}
+private void onResize(Event e) {
+       if (objIOleInPlaceActiveObject != null) {
+               RECT lpRect = new RECT();
+               OS.GetClientRect(handle, lpRect);
+               objIOleInPlaceActiveObject.ResizeBorder(lpRect, iOleInPlaceFrame.getAddress(), true);
+       }
+}
+private int QueryInterface(long riid, long ppvObject) {
+//     implements IUnknown, IOleInPlaceFrame, IOleContainer, IOleInPlaceUIWindow
+       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.IIDIOleInPlaceFrame) ) {
+               OS.MoveMemory(ppvObject, new long [] {iOleInPlaceFrame.getAddress()}, C.PTR_SIZEOF);
+               AddRef();
+               return COM.S_OK;
+       }
+
+       OS.MoveMemory(ppvObject, new long [] {0}, C.PTR_SIZEOF);
+       return COM.E_NOINTERFACE;
+}
+/**
+ * Decrement the count of references to this instance
+ *
+ * @return the current reference count
+ */
+int Release() {
+       refCount--;
+       if (refCount == 0){
+               disposeCOMInterfaces();
+               if (COM.FreeUnusedLibraries) {
+                       COM.CoFreeUnusedLibraries();
+               }
+       }
+       return refCount;
+}
+private void releaseObjectInterfaces() {
+       if (objIOleInPlaceActiveObject != null) {
+               objIOleInPlaceActiveObject.Release();
+       }
+       objIOleInPlaceActiveObject = null;
+}
+private int RemoveMenus(long hmenuShared) {
+
+       Menu menubar = getShell().getMenuBar();
+       if (menubar == null || menubar.isDisposed()) return COM.S_FALSE;
+
+       long hMenu = menubar.handle;
+
+       List<LONG> ids = new ArrayList<>();
+       if (this.fileMenuItems != null) {
+               for (int i = 0; i < this.fileMenuItems.length; i++) {
+                       MenuItem item = this.fileMenuItems[i];
+                       if (item != null && !item.isDisposed()) {
+                               int index = item.getParent().indexOf(item);
+                               // get Id from original menubar
+                               long id = getMenuItemID(hMenu, index);
+                               ids.add(new LONG(id));
+                       }
+               }
+       }
+       if (this.containerMenuItems != null) {
+               for (int i = 0; i < this.containerMenuItems.length; i++) {
+                       MenuItem item = this.containerMenuItems[i];
+                       if (item != null && !item.isDisposed()) {
+                               int index = item.getParent().indexOf(item);
+                               long id = getMenuItemID(hMenu, index);
+                               ids.add(new LONG(id));
+                       }
+               }
+       }
+       if (this.windowMenuItems != null) {
+               for (int i = 0; i < this.windowMenuItems.length; i++) {
+                       MenuItem item = this.windowMenuItems[i];
+                       if (item != null && !item.isDisposed()) {
+                               int index = item.getParent().indexOf(item);
+                               long id = getMenuItemID(hMenu, index);
+                               ids.add(new LONG(id));
+                       }
+               }
+       }
+       int index = OS.GetMenuItemCount(hmenuShared) - 1;
+       for (int i = index; i >= 0; i--) {
+               long id = getMenuItemID(hmenuShared, i);
+               if (ids.contains(new LONG(id))){
+                       OS.RemoveMenu(hmenuShared, i, OS.MF_BYPOSITION);
+               }
+       }
+       return COM.S_OK;
+}
+private int RequestBorderSpace(long pborderwidths) {
+       return COM.S_OK;
+}
+int SetActiveObject(long pActiveObject, long pszObjName) {
+       if (objIOleInPlaceActiveObject != null) {
+               objIOleInPlaceActiveObject.Release();
+               objIOleInPlaceActiveObject = null;
+       }
+       if (pActiveObject != 0) {
+               objIOleInPlaceActiveObject = new IOleInPlaceActiveObject(pActiveObject);
+               objIOleInPlaceActiveObject.AddRef();
+       }
+
+       return COM.S_OK;
+}
+private int SetBorderSpace(long pborderwidths) {
+       // A Control/Document can :
+       // Use its own toolbars, requesting border space of a specific size, or,
+       // Use no toolbars, but force the container to remove its toolbars by passing a
+       //   valid BORDERWIDTHS structure containing nothing but zeros in the pborderwidths parameter, or,
+       // Use no toolbars but allow the in-place container to leave its toolbars up by
+       //   passing NULL as the pborderwidths parameter.
+       if (objIOleInPlaceActiveObject == null) return COM.S_OK;
+       RECT borderwidth = new RECT();
+       if (pborderwidths == 0 || currentdoc == null ) return COM.S_OK;
+
+       COM.MoveMemory(borderwidth, pborderwidths, RECT.sizeof);
+       currentdoc.setBorderSpace(borderwidth);
+
+       return COM.S_OK;
+}
+/**
+ *
+ * Specify the menu items that should appear in the Container location when an OLE Document
+ * is in-place activated.
+ *
+ * <p>When an OLE Document is in-place active, the Document provides its own menus but the application
+ * is given the opportunity to merge some of its menus into the menubar.  The application
+ * is allowed to insert its menus in three locations: File (far left), Container(middle) and Window
+ * (far right just before Help).  The OLE Document retains control of the Edit, Object and Help
+ * menu locations.  Note that an application can insert more than one menu into a single location.
+ *
+ * <p>This method must be called before in place activation of the OLE Document.  After the Document
+ * is activated, the menu bar will not be modified until a subsequent activation.
+ *
+ * @param containerMenus an array of top level MenuItems to be inserted into the Container location of
+ *        the menubar
+ */
+public void setContainerMenus(MenuItem[] containerMenus){
+       containerMenuItems = containerMenus;
+}
+OleClientSite getCurrentDocument() {
+       return currentdoc;
+}
+void setCurrentDocument(OleClientSite doc) {
+       currentdoc = doc;
+
+       if (currentdoc != null && objIOleInPlaceActiveObject != null) {
+               RECT lpRect = new RECT();
+               OS.GetClientRect(handle, lpRect);
+               objIOleInPlaceActiveObject.ResizeBorder(lpRect, iOleInPlaceFrame.getAddress(), true);
+       }
+}
+/**
+ *
+ * Specify the menu items that should appear in the File location when an OLE Document
+ * is in-place activated.
+ *
+ * <p>When an OLE Document is in-place active, the Document provides its own menus but the application
+ * is given the opportunity to merge some of its menus into the menubar.  The application
+ * is allowed to insert its menus in three locations: File (far left), Container(middle) and Window
+ * (far right just before Help).  The OLE Document retains control of the Edit, Object and Help
+ * menu locations.  Note that an application can insert more than one menu into a single location.
+ *
+ * <p>This method must be called before in place activation of the OLE Document.  After the Document
+ * is activated, the menu bar will not be modified until a subsequent activation.
+ *
+ * @param fileMenus an array of top level MenuItems to be inserted into the File location of
+ *        the menubar
+ */
+public void setFileMenus(MenuItem[] fileMenus){
+       fileMenuItems = fileMenus;
+}
+private int SetMenu(long hmenuShared, long holemenu, long hwndActiveObject) {
+       long inPlaceActiveObject = 0;
+       if (objIOleInPlaceActiveObject != null)
+               inPlaceActiveObject = objIOleInPlaceActiveObject.getAddress();
+
+       Menu menubar = getShell().getMenuBar();
+       if (menubar == null || menubar.isDisposed()){
+               return COM.OleSetMenuDescriptor(0, getShell().handle, hwndActiveObject, iOleInPlaceFrame.getAddress(), inPlaceActiveObject);
+       }
+
+       long handle = menubar.getShell().handle;
+
+       if (hmenuShared == 0 && holemenu == 0) {
+               // re-instate the original menu - this occurs on deactivation
+               hmenuShared = menubar.handle;
+       }
+       if (hmenuShared == 0) return COM.E_FAIL;
+
+       shellHandle = handle;
+       oldMenuHandle = menubar.handle;
+       newMenuHandle = hmenuShared;
+       lastActivatedMenuHandle = newMenuHandle;
+
+       return COM.OleSetMenuDescriptor(holemenu, handle, hwndActiveObject, iOleInPlaceFrame.getAddress(), inPlaceActiveObject);
+}
+/**
+ *
+ * Set the menu items that should appear in the Window location when an OLE Document
+ * is in-place activated.
+ *
+ * <p>When an OLE Document is in-place active, the Document provides its own menus but the application
+ * is given the opportunity to merge some of its menus into the menubar.  The application
+ * is allowed to insert its menus in three locations: File (far left), Container(middle) and Window
+ * (far right just before Help).  The OLE Document retains control of the Edit, Object and Help
+ * menu locations.  Note that an application can insert more than one menu into a single location.
+ *
+ * <p>This method must be called before in place activation of the OLE Document.  After the Document
+ * is activated, the menu bar will not be modified until a subsequent activation.
+ *
+ * @param windowMenus an array of top level MenuItems to be inserted into the Window location of
+ *        the menubar
+ */
+public void setWindowMenus(MenuItem[] windowMenus){
+       windowMenuItems = windowMenus;
+}
+private boolean translateOleAccelerator(MSG msg) {
+       if (objIOleInPlaceActiveObject == null) return false;
+       int result = objIOleInPlaceActiveObject.TranslateAccelerator(msg);
+       return (result != COM.S_FALSE && result != COM.E_NOTIMPL);
+}
+private int TranslateAccelerator(long lpmsg, int wID){
+       Menu menubar = getShell().getMenuBar();
+       if (menubar == null || menubar.isDisposed() || !menubar.isEnabled()) return COM.S_FALSE;
+       if (wID < 0) return COM.S_FALSE;
+
+       Shell shell = menubar.getShell();
+       long hwnd = shell.handle;
+       long hAccel = OS.SendMessage(hwnd, OS.WM_APP+1, 0, 0);
+       if (hAccel == 0) return COM.S_FALSE;
+
+       MSG msg = new MSG();
+       OS.MoveMemory(msg, lpmsg, MSG.sizeof);
+       int result = OS.TranslateAccelerator(hwnd, hAccel, msg);
+       return result == 0 ? COM.S_FALSE : COM.S_OK;
+}
+}