diff org.eclipse.swt.win32.win32.x86/src/org/eclipse/swt/ole/win32/OleFrame.d @ 0:6dd524f61e62

add dwt win and basic java stuff
author Frank Benoit <benoit@tionex.de>
date Mon, 02 Mar 2009 14:44:16 +0100
parents
children 43904fec5dca
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/org.eclipse.swt.win32.win32.x86/src/org/eclipse/swt/ole/win32/OleFrame.d	Mon Mar 02 14:44:16 2009 +0100
@@ -0,0 +1,845 @@
+/*******************************************************************************
+ * Copyright (c) 2000, 2008 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
+ * Port to the D programming language:
+ *     Frank Benoit <benoit@tionex.de>
+ *******************************************************************************/
+module org.eclipse.swt.ole.win32.OleFrame;
+
+import org.eclipse.swt.SWT;
+import org.eclipse.swt.SWTException;
+import org.eclipse.swt.internal.ole.win32.COM;
+import org.eclipse.swt.internal.ole.win32.OLEIDL;
+import org.eclipse.swt.internal.ole.win32.extras;
+import org.eclipse.swt.internal.win32.OS;
+import org.eclipse.swt.widgets.Composite;
+import org.eclipse.swt.widgets.Control;
+import org.eclipse.swt.widgets.Display;
+import org.eclipse.swt.widgets.Event;
+import org.eclipse.swt.widgets.Listener;
+import org.eclipse.swt.widgets.Menu;
+import org.eclipse.swt.widgets.MenuItem;
+import org.eclipse.swt.widgets.Shell;
+import org.eclipse.swt.widgets.Widget;
+
+import org.eclipse.swt.ole.win32.OleClientSite;
+import org.eclipse.swt.ole.win32.OLE;
+
+import java.lang.all;
+import java.lang.Runnable;
+import tango.core.Array;
+
+/**
+ *
+ * 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 : Composite
+{
+    // Interfaces for this Ole Client Container
+    //private COMObject iUnknown;
+    private _IOleInPlaceFrameImpl 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 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 bool ignoreNextKey;
+    private static const short [] ACCENTS = [ cast(short)'~', '`', '\'', '^', '"'];
+
+    private static const String CONSUME_KEY = "org.eclipse.swt.OleFrame.ConsumeKey"; //$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 this(Composite parent, int style) {
+    super(parent, style);
+
+    createCOMInterfaces();
+
+    // setup cleanup proc
+    listener = new class() Listener  {
+        public void handleEvent(Event 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 (Display display_) {
+    if (display_.getData(CHECK_FOCUS) !is null) return;
+    display_.setData(CHECK_FOCUS, new ArrayWrapperString(CHECK_FOCUS));
+    static const int time = 50;
+    auto timer = new class(display_) Runnable {
+        Display display;
+        Control[1] lastFocus;
+        this( Display display){ this.display = display; }
+        public void run() {
+            if (( null !is cast(OleClientSite)lastFocus[0] ) && !lastFocus[0].isDisposed()) {
+                // ignore popup menus and dialogs
+                auto hwnd = OS.GetFocus();
+                while (hwnd !is null) {
+                    auto ownerHwnd = OS.GetWindow(hwnd, OS.GW_OWNER);
+                    if (ownerHwnd !is null) {
+                        display.timerExec(time, this);
+                        return;
+                    }
+                    hwnd = OS.GetParent(hwnd);
+                }
+            }
+            if (lastFocus[0] is null || lastFocus[0].isDisposed() || !lastFocus[0].isFocusControl()) {
+                Control currentFocus = display.getFocusControl();
+                if ( auto frame = cast(OleFrame)currentFocus ) {
+                    currentFocus = frame.getCurrentDocument();
+                }
+                if (lastFocus[0] !is currentFocus) {
+                    Event event = new Event();
+                    if (( null !is cast(OleClientSite)lastFocus[0] ) && !lastFocus[0].isDisposed()) {
+                        lastFocus[0].notifyListeners (SWT.FocusOut, event);
+                    }
+                    if (( null !is cast(OleClientSite)currentFocus ) && !currentFocus.isDisposed()) {
+                        currentFocus.notifyListeners(SWT.FocusIn, event);
+                    }
+                }
+                lastFocus[0] = currentFocus;
+            }
+            display.timerExec(time, this);
+        }
+    };
+    display_.timerExec(time, timer);
+}
+private static void initMsgHook(Display display) {
+    if (display.getData(HHOOK) !is null) return;
+    //final Callback callback = new Callback(OleFrame.class, "getMsgProc", 3); //$NON-NLS-1$
+    //int address = callback.getAddress();
+    //if (address is 0) SWT.error(SWT.ERROR_NO_MORE_CALLBACKS);
+    int threadId = OS.GetCurrentThreadId();
+    auto hHook_ = OS.SetWindowsHookEx(OS.WH_GETMESSAGE, &getMsgProc, null, threadId);
+    if (hHook_ is null) {
+        //callback.dispose();
+        return;
+    }
+    display.setData(HHOOK, new ValueWrapperT!(void*)(hHook_));
+    display.setData(HHOOKMSG, new ValueWrapperT!(MSG*)(new MSG()));
+    display.disposeExec(new class(hHook_) Runnable {
+        void* hHook;
+        this( void* hHook ){ this.hHook = hHook; }
+        public void run() {
+            if (hHook !is null) OS.UnhookWindowsHookEx(hHook);
+            //if (callback !is null) callback.dispose();
+        }
+    });
+}
+static extern(Windows) int getMsgProc(int code, uint wParam, int lParam) {
+    Display display = Display.getCurrent();
+    if (display is null) return 0;
+    auto hHook = cast(ValueWrapperT!(void*))display.getData(HHOOK);
+    if (hHook is null) return 0;
+    if (code < 0) {
+        return OS.CallNextHookEx(hHook.value, code, wParam, lParam);
+    }
+    MSG* msg = cast(MSG*)(cast(ValueWrapperT!(MSG*))display.getData(HHOOKMSG)).value;
+    OS.MoveMemory(msg, lParam, MSG.sizeof);
+    int message = msg.message;
+    if (OS.WM_KEYFIRST <= message && message <= OS.WM_KEYLAST) {
+        if (display !is null) {
+            Widget widget = null;
+            auto hwnd = msg.hwnd;
+            while (hwnd !is null) {
+                widget = display.findWidget (hwnd);
+                if (widget !is null) break;
+                hwnd = OS.GetParent (hwnd);
+            }
+            if (widget !is null && (null !is cast(OleClientSite)widget )) {
+                OleClientSite site = cast(OleClientSite)widget;
+                if (site.handle is hwnd) {
+                    bool 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;
+                    bool rc = cast(bool) OS.GetGUIThreadInfo(thread, lpgui);
+                    int mask = OS.GUI_INMENUMODE | OS.GUI_INMOVESIZE | OS.GUI_POPUPMENUMODE | OS.GUI_SYSTEMMENUMODE;
+                    if (!rc || (lpgui.flags & mask) is 0) {
+                        OleFrame frame = site.frame;
+                        frame.setData(CONSUME_KEY, null);
+                        consumed = frame.translateOleAccelerator(msg);
+                        if (frame.getData(CONSUME_KEY) !is null) consumed = false;
+                        frame.setData(CONSUME_KEY, null);
+                    }
+                    bool accentKey = false;
+                    switch (msg.message) {
+                        case OS.WM_KEYDOWN:
+                        case OS.WM_SYSKEYDOWN: {
+                            static if (!OS.IsWinCE) {
+                                switch (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: {
+                                        /*
+                                        * Bug in Windows. The high bit in the result of MapVirtualKey() on
+                                        * Windows NT is bit 32 while the high bit on Windows 95 is bit 16.
+                                        * They should both be bit 32.  The fix is to test the right bit.
+                                        */
+                                        int mapKey = OS.MapVirtualKey (msg.wParam, 2);
+                                        if (mapKey !is 0) {
+                                            accentKey = (mapKey & (OS.IsWinNT ? 0x80000000 : 0x8000)) !is 0;
+                                            if (!accentKey) {
+                                                for (int i=0; i<ACCENTS.length; i++) {
+                                                    int value = OS.VkKeyScan (ACCENTS [i]);
+                                                    if (value !is -1 && (value & 0xFF) is msg.wParam) {
+                                                        int state = value >> 8;
+                                                        if ((OS.GetKeyState (OS.VK_SHIFT) < 0) is ((state & 0x1) !is 0) &&
+                                                            (OS.GetKeyState (OS.VK_CONTROL) < 0) is ((state & 0x2) !is 0) &&
+                                                            (OS.GetKeyState (OS.VK_MENU) < 0) is ((state & 0x4) !is 0)) {
+                                                                if ((state & 0x7) !is 0) accentKey = true;
+                                                                break;
+                                                        }
+                                                    }
+                                                }
+                                            }
+                                        }
+                                        break;
+                                    }
+                                }
+                            }
+                            break;
+                        }
+                        default:
+                    }
+                    /* Allow OleClientSite to process key events before activeX control */
+                    if (!consumed && !accentKey && !ignoreNextKey) {
+                        auto hwndOld = msg.hwnd;
+                        msg.hwnd = site.handle;
+                        consumed = OS.DispatchMessage (msg) is 1;
+                        msg.hwnd = hwndOld;
+                    }
+                    switch (msg.message) {
+                        case OS.WM_KEYDOWN:
+                        case OS.WM_SYSKEYDOWN: {
+                            switch (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;
+                                }
+                            }
+                        }
+                        default:
+                    }
+
+                    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, 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 _IOleInPlaceFrameImpl(this);
+}
+private void disposeCOMInterfaces () {
+    iOleInPlaceFrame = null;
+}
+private HRESULT GetBorder(LPRECT 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 is null) return COM.E_INVALIDARG;
+    RECT* rectBorder = new RECT();
+    // Coordinates must be relative to the window
+    OS.GetClientRect(handle, lprectBorder);
+    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;
+}
+IOleInPlaceFrame getIOleInPlaceFrame() {
+    return iOleInPlaceFrame;
+}
+private int getMenuItemID(HMENU hMenu, int index) {
+    int id = 0;
+    MENUITEMINFO lpmii;
+    lpmii.cbSize = OS.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) is OS.MF_POPUP) {
+        id = cast(int)lpmii.hSubMenu;
+    } else {
+        id = lpmii.wID;
+    }
+    return id;
+}
+private int GetWindow(HWND* phwnd) {
+    if (phwnd !is null) {
+        *phwnd = handle;
+    }
+    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 HRESULT InsertMenus( HMENU hmenuShared, LPOLEMENUGROUPWIDTHS lpMenuWidths ) {
+    // locate menu bar
+    Menu menubar = getShell().getMenuBar();
+    if (menubar is null || menubar.isDisposed()) {
+        int temp = 0;
+        COM.MoveMemory(lpMenuWidths, &temp, 4);
+        return COM.S_OK;
+    }
+    HMENU 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;
+    auto hHeap = OS.GetProcessHeap();
+    int cch = 128;
+    int byteCount = cch * TCHAR.sizeof;
+    auto pszText = cast(TCHAR*) OS.HeapAlloc(hHeap, OS.HEAP_ZERO_MEMORY, byteCount);
+    lpmii.cbSize = OS.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 !is null) {
+        for (int i = 0; i < this.fileMenuItems.length; i++) {
+            MenuItem item = this.fileMenuItems[i];
+            if (item !is 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
+    COM.MoveMemory(lpMenuWidths, &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 !is null) {
+        for (int i = 0; i < this.containerMenuItems.length; i++) {
+            MenuItem item = this.containerMenuItems[i];
+            if (item !is 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
+    COM.MoveMemory((cast(void*)lpMenuWidths) + 8, &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 !is null) {
+        for (int i = 0; i < this.windowMenuItems.length; i++) {
+            MenuItem item = this.windowMenuItems[i];
+            if (item !is 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
+    COM.MoveMemory((cast(void*)lpMenuWidths) + 16, &windowMenuCount, 4);
+
+    // free resources used in querying the OS
+    if (pszText !is null)
+        OS.HeapFree(hHeap, 0, pszText);
+    return COM.S_OK;
+}
+void onActivate(Event e) {
+    if (objIOleInPlaceActiveObject !is null) {
+        objIOleInPlaceActiveObject.OnFrameWindowActivate(true);
+    }
+    if (objIOleInPlaceActiveObject !is null) {
+        objIOleInPlaceActiveObject.OnDocWindowActivate(true);
+    }
+}
+void onDeactivate(Event e) {
+    if (objIOleInPlaceActiveObject !is null) {
+        objIOleInPlaceActiveObject.OnFrameWindowActivate(false);
+    }
+    if (objIOleInPlaceActiveObject !is null) {
+        objIOleInPlaceActiveObject.OnDocWindowActivate(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);
+}
+private void onResize(Event e) {
+    if (objIOleInPlaceActiveObject !is null) {
+        RECT lpRect;
+        OS.GetClientRect(handle, &lpRect);
+        objIOleInPlaceActiveObject.ResizeBorder(&lpRect, iOleInPlaceFrame, true);
+    }
+}
+private HRESULT QueryInterface(REFIID riid, void** ppvObject) {
+//  implements IUnknown, IOleInPlaceFrame, IOleContainer, IOleInPlaceUIWindow
+    if (riid is null || ppvObject is null)
+        return COM.E_INVALIDARG;
+
+    if (COM.IsEqualGUID(riid, &COM.IIDIUnknown) || COM.IsEqualGUID(riid, &COM.IIDIOleInPlaceFrame) ) {
+        *ppvObject = cast(void*)cast(IOleInPlaceFrame)iOleInPlaceFrame;
+        AddRef();
+        return COM.S_OK;
+    }
+
+    *ppvObject = null;
+    return COM.E_NOINTERFACE;
+}
+/**
+ * Decrement the count of references to this instance
+ *
+ * @return the current reference count
+ */
+int Release() {
+    refCount--;
+    if (refCount is 0){
+        disposeCOMInterfaces();
+        COM.CoFreeUnusedLibraries();
+    }
+    return refCount;
+}
+private void releaseObjectInterfaces() {
+    if (objIOleInPlaceActiveObject !is null) {
+        objIOleInPlaceActiveObject.Release();
+    }
+    objIOleInPlaceActiveObject = null;
+}
+private int RemoveMenus(HMENU hmenuShared) {
+
+    Menu menubar = getShell().getMenuBar();
+    if (menubar is null || menubar.isDisposed()) return COM.S_FALSE;
+
+    auto hMenu = menubar.handle;
+
+    int[] ids;
+    if (this.fileMenuItems !is null) {
+        for (int i = 0; i < this.fileMenuItems.length; i++) {
+            MenuItem item = this.fileMenuItems[i];
+            if (item !is null && !item.isDisposed()) {
+                int index = item.getParent().indexOf(item);
+                // get Id from original menubar
+                int id = getMenuItemID(hMenu, index);
+                ids ~= id;
+            }
+        }
+    }
+    if (this.containerMenuItems !is null) {
+        for (int i = 0; i < this.containerMenuItems.length; i++) {
+            MenuItem item = this.containerMenuItems[i];
+            if (item !is null && !item.isDisposed()) {
+                int index = item.getParent().indexOf(item);
+                int id = getMenuItemID(hMenu, index);
+                ids ~= id;
+            }
+        }
+    }
+    if (this.windowMenuItems !is null) {
+        for (int i = 0; i < this.windowMenuItems.length; i++) {
+            MenuItem item = this.windowMenuItems[i];
+            if (item !is null && !item.isDisposed()) {
+                int index = item.getParent().indexOf(item);
+                int id = getMenuItemID(hMenu, index);
+                ids ~= id;
+            }
+        }
+    }
+    int index = OS.GetMenuItemCount(hmenuShared) - 1;
+    for (int i = index; i >= 0; i--) {
+        int id = getMenuItemID(hmenuShared, i);
+        if ( ids.contains(id)){
+            OS.RemoveMenu(hmenuShared, i, OS.MF_BYPOSITION);
+        }
+    }
+    return COM.S_OK;
+}
+private int RequestBorderSpace(LPCBORDERWIDTHS pborderwidths) {
+    return COM.S_OK;
+}
+HRESULT SetActiveObject( LPOLEINPLACEACTIVEOBJECT pActiveObject, LPCOLESTR pszObjName ) {
+    if (objIOleInPlaceActiveObject !is null) {
+        objIOleInPlaceActiveObject.Release();
+        objIOleInPlaceActiveObject = null;
+    }
+    if (pActiveObject !is null) {
+        objIOleInPlaceActiveObject = pActiveObject;
+        objIOleInPlaceActiveObject.AddRef();
+    }
+
+    return COM.S_OK;
+}
+
+private HRESULT SetBorderSpace( LPCBORDERWIDTHS 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 is null) return COM.S_OK;
+    RECT* borderwidth = new RECT();
+    if (pborderwidths is null || currentdoc is 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 !is null && objIOleInPlaceActiveObject !is null) {
+        RECT lpRect;
+        OS.GetClientRect(handle, &lpRect);
+        objIOleInPlaceActiveObject.ResizeBorder(&lpRect, iOleInPlaceFrame, 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;
+}
+HRESULT SetMenu( HMENU hmenuShared, HOLEMENU holemenu, HWND hwndActiveObject ) {
+    IOleInPlaceActiveObject inPlaceActiveObject;
+    if (objIOleInPlaceActiveObject !is null)
+        inPlaceActiveObject = objIOleInPlaceActiveObject;
+
+    Menu menubar = getShell().getMenuBar();
+    if (menubar is null || menubar.isDisposed()){
+        return COM.OleSetMenuDescriptor(null, getShell().handle, hwndActiveObject, iOleInPlaceFrame, inPlaceActiveObject);
+    }
+
+    HWND handle = menubar.getShell().handle;
+
+    if (hmenuShared is null && holemenu is null) {
+        // re-instate the original menu - this occurs on deactivation
+        hmenuShared = menubar.handle;
+    }
+    if (hmenuShared is null) return COM.E_FAIL;
+
+    OS.SetMenu(handle, hmenuShared);
+    OS.DrawMenuBar(handle);
+
+    return COM.OleSetMenuDescriptor(holemenu, handle, hwndActiveObject, iOleInPlaceFrame, 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 bool translateOleAccelerator(MSG* msg) {
+    if (objIOleInPlaceActiveObject is null) return false;
+    int result = objIOleInPlaceActiveObject.TranslateAccelerator(msg);
+    return (result != COM.S_FALSE && result != COM.E_NOTIMPL);
+}
+
+HRESULT TranslateAccelerator( LPMSG lpmsg, WORD wID ){
+    Menu menubar = getShell().getMenuBar();
+    if (menubar is null || menubar.isDisposed() || !menubar.isEnabled()) return COM.S_FALSE;
+    if (wID < 0) return COM.S_FALSE;
+
+    Shell shell = menubar.getShell();
+    HWND hwnd = shell.handle;
+    HACCEL hAccel = cast(HACCEL)OS.SendMessage(hwnd, OS.WM_APP+1, 0, 0);
+    if (hAccel is null) return COM.S_FALSE;
+
+    MSG msg = *lpmsg;
+    int result = OS.TranslateAccelerator(hwnd, hAccel, &msg);
+    return result == 0 ? COM.S_FALSE : COM.S_OK;
+}
+}
+
+//  implements IOleInPlaceFrame, IOleInPlaceUIWindow, IOleWindow, IUnknown
+class _IOleInPlaceFrameImpl : IOleInPlaceFrame {
+
+    OleFrame parent;
+    this(OleFrame p) { parent = p; }
+extern (Windows) :
+    // interface of IUnknown
+    HRESULT QueryInterface(REFIID riid, void ** ppvObject) { return parent.QueryInterface(riid, ppvObject); }
+    ULONG AddRef()  { return parent.AddRef(); }
+    ULONG Release() { return parent.Release(); }
+
+    // interface IOleWindow
+    HRESULT GetWindow( HWND * phwnd ) { return parent.GetWindow(phwnd); }
+    HRESULT ContextSensitiveHelp( BOOL fEnterMode ){ return COM.S_OK; }
+
+    //interface IOleInPlaceUIWindow
+    HRESULT GetBorder( LPRECT lprectBorder ) { return parent.GetBorder(lprectBorder); }
+    HRESULT RequestBorderSpace( LPCBORDERWIDTHS pborderwidths ){ return COM.S_OK; }
+    HRESULT SetBorderSpace( LPCBORDERWIDTHS pborderwidths ) { return parent.SetBorderSpace(pborderwidths); }
+    HRESULT SetActiveObject( LPOLEINPLACEACTIVEOBJECT pActiveObject, LPCOLESTR pszObjName ) {
+        return parent.SetActiveObject(pActiveObject, pszObjName);
+    }
+
+    // interface IOleInPlaceFrame : IOleInPlaceUIWindow
+    HRESULT InsertMenus( HMENU hmenuShared, LPOLEMENUGROUPWIDTHS lpMenuWidths ){
+        return parent.InsertMenus(hmenuShared, lpMenuWidths);
+    }
+    HRESULT SetMenu( HMENU hmenuShared, HOLEMENU holemenu, HWND hwndActiveObject ){
+        return parent.SetMenu(hmenuShared, holemenu, hwndActiveObject);
+    }
+    HRESULT RemoveMenus( HMENU hmenuShared ) {
+        return parent.RemoveMenus(hmenuShared);
+    }
+    HRESULT SetStatusText( LPCOLESTR pszStatusText ) { return COM.E_NOTIMPL; }
+    HRESULT EnableModeless( BOOL fEnable ) { return COM.E_NOTIMPL; }
+    HRESULT TranslateAccelerator( LPMSG lpmsg, WORD wID ) {
+        return parent.TranslateAccelerator(lpmsg, wID);
+    }
+
+}
+