| /* |
| * Copyright 2005 Jacek Caban |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include "config.h" |
| |
| #include <stdarg.h> |
| #include <stdio.h> |
| |
| #define COBJMACROS |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "winuser.h" |
| #include "ole2.h" |
| |
| #include "wine/debug.h" |
| |
| #include "mshtml_private.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(mshtml); |
| |
| typedef struct { |
| IOleUndoManager IOleUndoManager_iface; |
| |
| LONG ref; |
| } UndoManager; |
| |
| static inline UndoManager *impl_from_IOleUndoManager(IOleUndoManager *iface) |
| { |
| return CONTAINING_RECORD(iface, UndoManager, IOleUndoManager_iface); |
| } |
| |
| static HRESULT WINAPI OleUndoManager_QueryInterface(IOleUndoManager *iface, REFIID riid, void **ppv) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| |
| if(IsEqualGUID(riid, &IID_IUnknown)) { |
| TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv); |
| *ppv = &This->IOleUndoManager_iface; |
| }else if(IsEqualGUID(riid, &IID_IOleUndoManager)) { |
| TRACE("(%p)->(IID_IOleUndoManager %p)\n", This, ppv); |
| *ppv = &This->IOleUndoManager_iface; |
| }else { |
| *ppv = NULL; |
| FIXME("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv); |
| return E_NOINTERFACE; |
| } |
| |
| IUnknown_AddRef((IUnknown*)*ppv); |
| return S_OK; |
| } |
| |
| static ULONG WINAPI OleUndoManager_AddRef(IOleUndoManager *iface) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| LONG ref = InterlockedIncrement(&This->ref); |
| |
| TRACE("(%p) ref=%d\n", This, ref); |
| |
| return ref; |
| } |
| |
| static ULONG WINAPI OleUndoManager_Release(IOleUndoManager *iface) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| LONG ref = InterlockedDecrement(&This->ref); |
| |
| TRACE("(%p) ref=%d\n", This, ref); |
| |
| if(!ref) |
| heap_free(This); |
| |
| return ref; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_Open(IOleUndoManager *iface, IOleParentUndoUnit *pPUU) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, pPUU); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_Close(IOleUndoManager *iface, IOleParentUndoUnit *pPUU, |
| BOOL fCommit) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p %x)\n", This, pPUU, fCommit); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_Add(IOleUndoManager *iface, IOleUndoUnit *pUU) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, pUU); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_GetOpenParentState(IOleUndoManager *iface, DWORD *pdwState) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, pdwState); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_DiscardFrom(IOleUndoManager *iface, IOleUndoUnit *pUU) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, pUU); |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_UndoTo(IOleUndoManager *iface, IOleUndoUnit *pUU) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, pUU); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_RedoTo(IOleUndoManager *iface, IOleUndoUnit *pUU) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, pUU); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_EnumUndoable(IOleUndoManager *iface, |
| IEnumOleUndoUnits **ppEnum) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, ppEnum); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_EnumRedoable(IOleUndoManager *iface, |
| IEnumOleUndoUnits **ppEnum) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, ppEnum); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_GetLastUndoDescription(IOleUndoManager *iface, BSTR *pBstr) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, pBstr); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_GetLastRedoDescription(IOleUndoManager *iface, BSTR *pBstr) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%p)\n", This, pBstr); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI OleUndoManager_Enable(IOleUndoManager *iface, BOOL fEnable) |
| { |
| UndoManager *This = impl_from_IOleUndoManager(iface); |
| FIXME("(%p)->(%x)\n", This, fEnable); |
| return E_NOTIMPL; |
| } |
| |
| static const IOleUndoManagerVtbl OleUndoManagerVtbl = { |
| OleUndoManager_QueryInterface, |
| OleUndoManager_AddRef, |
| OleUndoManager_Release, |
| OleUndoManager_Open, |
| OleUndoManager_Close, |
| OleUndoManager_Add, |
| OleUndoManager_GetOpenParentState, |
| OleUndoManager_DiscardFrom, |
| OleUndoManager_UndoTo, |
| OleUndoManager_RedoTo, |
| OleUndoManager_EnumUndoable, |
| OleUndoManager_EnumRedoable, |
| OleUndoManager_GetLastUndoDescription, |
| OleUndoManager_GetLastRedoDescription, |
| OleUndoManager_Enable |
| }; |
| |
| static IOleUndoManager *create_undomgr(void) |
| { |
| UndoManager *ret = heap_alloc(sizeof(UndoManager)); |
| |
| if (!ret) return NULL; |
| |
| ret->IOleUndoManager_iface.lpVtbl = &OleUndoManagerVtbl; |
| ret->ref = 1; |
| |
| return &ret->IOleUndoManager_iface; |
| } |
| |
| typedef struct { |
| IHTMLEditServices IHTMLEditServices_iface; |
| LONG ref; |
| } editsvcs; |
| |
| static inline editsvcs *impl_from_IHTMLEditServices(IHTMLEditServices *iface) |
| { |
| return CONTAINING_RECORD(iface, editsvcs, IHTMLEditServices_iface); |
| } |
| |
| static HRESULT WINAPI editsvcs_QueryInterface(IHTMLEditServices *iface, REFIID riid, void **ppv) |
| { |
| editsvcs *This = impl_from_IHTMLEditServices(iface); |
| |
| TRACE("(%p)->(%s %p)\n", This, debugstr_mshtml_guid(riid), ppv); |
| |
| if(IsEqualGUID(riid, &IID_IUnknown)) { |
| *ppv = &This->IHTMLEditServices_iface; |
| } else if(IsEqualGUID(riid, &IID_IHTMLEditServices)) { |
| *ppv = &This->IHTMLEditServices_iface; |
| } else { |
| *ppv = NULL; |
| FIXME("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv); |
| return E_NOINTERFACE; |
| } |
| |
| IUnknown_AddRef((IUnknown*)*ppv); |
| return S_OK; |
| } |
| |
| static ULONG WINAPI editsvcs_AddRef(IHTMLEditServices *iface) |
| { |
| editsvcs *This = impl_from_IHTMLEditServices(iface); |
| LONG ref = InterlockedIncrement(&This->ref); |
| |
| TRACE("(%p) ref=%d\n", This, ref); |
| return ref; |
| } |
| |
| static ULONG WINAPI editsvcs_Release(IHTMLEditServices *iface) |
| { |
| editsvcs *This = impl_from_IHTMLEditServices(iface); |
| LONG ref = InterlockedDecrement(&This->ref); |
| |
| TRACE("(%p) ref=%d\n", This, ref); |
| |
| if(!ref) |
| heap_free(This); |
| |
| return ref; |
| } |
| |
| static HRESULT WINAPI editsvcs_AddDesigner(IHTMLEditServices *iface, |
| IHTMLEditDesigner *pIDesigner) |
| { |
| editsvcs *This = impl_from_IHTMLEditServices(iface); |
| FIXME("(%p)->(%p)\n", This, pIDesigner); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI editsvcs_RemoveDesigner(IHTMLEditServices *iface, |
| IHTMLEditDesigner *pIDesigner) |
| { |
| editsvcs *This = impl_from_IHTMLEditServices(iface); |
| FIXME("(%p)->(%p)\n", This, pIDesigner); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI editsvcs_GetSelectionServices(IHTMLEditServices *iface, |
| IMarkupContainer *pIContainer, ISelectionServices **ppSelSvc) |
| { |
| editsvcs *This = impl_from_IHTMLEditServices(iface); |
| FIXME("(%p)->(%p,%p)\n", This, pIContainer, ppSelSvc); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI editsvcs_MoveToSelectionAnchor(IHTMLEditServices *iface, |
| IMarkupPointer *pIStartAnchor) |
| { |
| editsvcs *This = impl_from_IHTMLEditServices(iface); |
| FIXME("(%p)->(%p)\n", This, pIStartAnchor); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI editsvcs_MoveToSelectionEnd(IHTMLEditServices *iface, |
| IMarkupPointer *pIEndAnchor) |
| { |
| editsvcs *This = impl_from_IHTMLEditServices(iface); |
| FIXME("(%p)->(%p)\n", This, pIEndAnchor); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI editsvcs_SelectRange(IHTMLEditServices *iface, |
| IMarkupPointer *pStart, IMarkupPointer *pEnd, SELECTION_TYPE eType) |
| { |
| editsvcs *This = impl_from_IHTMLEditServices(iface); |
| FIXME("(%p)->(%p,%p,%#x)\n", This, pStart, pEnd, eType); |
| return E_NOTIMPL; |
| } |
| |
| static const IHTMLEditServicesVtbl editsvcsVtbl = { |
| editsvcs_QueryInterface, |
| editsvcs_AddRef, |
| editsvcs_Release, |
| editsvcs_AddDesigner, |
| editsvcs_RemoveDesigner, |
| editsvcs_GetSelectionServices, |
| editsvcs_MoveToSelectionAnchor, |
| editsvcs_MoveToSelectionEnd, |
| editsvcs_SelectRange, |
| }; |
| |
| static IHTMLEditServices *create_editsvcs(void) |
| { |
| editsvcs *ret = heap_alloc(sizeof(*ret)); |
| |
| if (ret) { |
| ret->IHTMLEditServices_iface.lpVtbl = &editsvcsVtbl; |
| ret->ref = 1; |
| return &ret->IHTMLEditServices_iface; |
| } |
| |
| return NULL; |
| } |
| |
| /********************************************************** |
| * IServiceProvider implementation |
| */ |
| |
| static inline HTMLDocument *impl_from_IServiceProvider(IServiceProvider *iface) |
| { |
| return CONTAINING_RECORD(iface, HTMLDocument, IServiceProvider_iface); |
| } |
| |
| static HRESULT WINAPI ServiceProvider_QueryInterface(IServiceProvider *iface, REFIID riid, void **ppv) |
| { |
| HTMLDocument *This = impl_from_IServiceProvider(iface); |
| return htmldoc_query_interface(This, riid, ppv); |
| } |
| |
| static ULONG WINAPI ServiceProvider_AddRef(IServiceProvider *iface) |
| { |
| HTMLDocument *This = impl_from_IServiceProvider(iface); |
| return htmldoc_addref(This); |
| } |
| |
| static ULONG WINAPI ServiceProvider_Release(IServiceProvider *iface) |
| { |
| HTMLDocument *This = impl_from_IServiceProvider(iface); |
| return htmldoc_release(This); |
| } |
| |
| static HRESULT WINAPI ServiceProvider_QueryService(IServiceProvider *iface, REFGUID guidService, |
| REFIID riid, void **ppv) |
| { |
| HTMLDocument *This = impl_from_IServiceProvider(iface); |
| |
| if(IsEqualGUID(&CLSID_CMarkup, guidService)) { |
| FIXME("(%p)->(CLSID_CMarkup %s %p)\n", This, debugstr_guid(riid), ppv); |
| return E_NOINTERFACE; |
| } |
| |
| if(IsEqualGUID(&SID_SOleUndoManager, guidService)) { |
| TRACE("SID_SOleUndoManager\n"); |
| |
| if(!This->doc_obj->undomgr) |
| This->doc_obj->undomgr = create_undomgr(); |
| |
| if (!This->doc_obj->undomgr) |
| return E_OUTOFMEMORY; |
| |
| return IOleUndoManager_QueryInterface(This->doc_obj->undomgr, riid, ppv); |
| } |
| |
| if(IsEqualGUID(&SID_SContainerDispatch, guidService)) { |
| TRACE("SID_SContainerDispatch\n"); |
| return IHTMLDocument2_QueryInterface(&This->IHTMLDocument2_iface, riid, ppv); |
| } |
| |
| if(IsEqualGUID(&IID_IWindowForBindingUI, guidService)) { |
| TRACE("IID_IWindowForBindingUI\n"); |
| return IWindowForBindingUI_QueryInterface(&This->doc_obj->IWindowForBindingUI_iface, riid, ppv); |
| } |
| |
| if(IsEqualGUID(&SID_SHTMLEditServices, guidService)) { |
| TRACE("SID_SHTMLEditServices\n"); |
| |
| if(!This->doc_obj->editsvcs) |
| This->doc_obj->editsvcs = create_editsvcs(); |
| |
| if (!This->doc_obj->editsvcs) |
| return E_OUTOFMEMORY; |
| |
| return IHTMLEditServices_QueryInterface(This->doc_obj->editsvcs, riid, ppv); |
| } |
| |
| TRACE("(%p)->(%s %s %p)\n", This, debugstr_guid(guidService), debugstr_guid(riid), ppv); |
| |
| if(This->doc_obj->client) { |
| HRESULT hres; |
| |
| hres = do_query_service((IUnknown*)This->doc_obj->client, guidService, riid, ppv); |
| if(SUCCEEDED(hres)) |
| return hres; |
| } |
| |
| FIXME("unknown service %s\n", debugstr_guid(guidService)); |
| return E_NOINTERFACE; |
| } |
| |
| static const IServiceProviderVtbl ServiceProviderVtbl = { |
| ServiceProvider_QueryInterface, |
| ServiceProvider_AddRef, |
| ServiceProvider_Release, |
| ServiceProvider_QueryService |
| }; |
| |
| void HTMLDocument_Service_Init(HTMLDocument *This) |
| { |
| This->IServiceProvider_iface.lpVtbl = &ServiceProviderVtbl; |
| } |