|  | /* | 
|  | *  ITfDocumentMgr implementation | 
|  | * | 
|  | *  Copyright 2009 Aric Stewart, CodeWeavers | 
|  | * | 
|  | * 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> | 
|  |  | 
|  | #define COBJMACROS | 
|  |  | 
|  | #include "wine/debug.h" | 
|  | #include "windef.h" | 
|  | #include "winbase.h" | 
|  | #include "winreg.h" | 
|  | #include "winuser.h" | 
|  | #include "shlwapi.h" | 
|  | #include "winerror.h" | 
|  | #include "objbase.h" | 
|  |  | 
|  | #include "wine/unicode.h" | 
|  |  | 
|  | #include "msctf.h" | 
|  | #include "msctf_internal.h" | 
|  |  | 
|  | WINE_DEFAULT_DEBUG_CHANNEL(msctf); | 
|  |  | 
|  | typedef struct tagDocumentMgr { | 
|  | ITfDocumentMgr ITfDocumentMgr_iface; | 
|  | ITfSource ITfSource_iface; | 
|  | LONG refCount; | 
|  |  | 
|  | /* Aggregation */ | 
|  | ITfCompartmentMgr  *CompartmentMgr; | 
|  |  | 
|  | ITfContext*  contextStack[2]; /* limit of 2 contexts */ | 
|  | ITfThreadMgrEventSink* ThreadMgrSink; | 
|  | } DocumentMgr; | 
|  |  | 
|  | typedef struct tagEnumTfContext { | 
|  | IEnumTfContexts IEnumTfContexts_iface; | 
|  | LONG refCount; | 
|  |  | 
|  | DWORD   index; | 
|  | DocumentMgr *docmgr; | 
|  | } EnumTfContext; | 
|  |  | 
|  | static HRESULT EnumTfContext_Constructor(DocumentMgr* mgr, IEnumTfContexts **ppOut); | 
|  |  | 
|  | static inline DocumentMgr *impl_from_ITfDocumentMgr(ITfDocumentMgr *iface) | 
|  | { | 
|  | return CONTAINING_RECORD(iface, DocumentMgr, ITfDocumentMgr_iface); | 
|  | } | 
|  |  | 
|  | static inline DocumentMgr *impl_from_ITfSource(ITfSource *iface) | 
|  | { | 
|  | return CONTAINING_RECORD(iface, DocumentMgr, ITfSource_iface); | 
|  | } | 
|  |  | 
|  | static inline EnumTfContext *impl_from_IEnumTfContexts(IEnumTfContexts *iface)\ | 
|  | { | 
|  | return CONTAINING_RECORD(iface, EnumTfContext, IEnumTfContexts_iface); | 
|  | } | 
|  |  | 
|  | static void DocumentMgr_Destructor(DocumentMgr *This) | 
|  | { | 
|  | ITfThreadMgr *tm; | 
|  | TRACE("destroying %p\n", This); | 
|  |  | 
|  | TF_GetThreadMgr(&tm); | 
|  | ThreadMgr_OnDocumentMgrDestruction(tm, &This->ITfDocumentMgr_iface); | 
|  |  | 
|  | if (This->contextStack[0]) | 
|  | ITfContext_Release(This->contextStack[0]); | 
|  | if (This->contextStack[1]) | 
|  | ITfContext_Release(This->contextStack[1]); | 
|  | CompartmentMgr_Destructor(This->CompartmentMgr); | 
|  | HeapFree(GetProcessHeap(),0,This); | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI DocumentMgr_QueryInterface(ITfDocumentMgr *iface, REFIID iid, LPVOID *ppvOut) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfDocumentMgr(iface); | 
|  | *ppvOut = NULL; | 
|  |  | 
|  | if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_ITfDocumentMgr)) | 
|  | { | 
|  | *ppvOut = This; | 
|  | } | 
|  | else if (IsEqualIID(iid, &IID_ITfSource)) | 
|  | { | 
|  | *ppvOut = &This->ITfSource_iface; | 
|  | } | 
|  | else if (IsEqualIID(iid, &IID_ITfCompartmentMgr)) | 
|  | { | 
|  | *ppvOut = This->CompartmentMgr; | 
|  | } | 
|  |  | 
|  | if (*ppvOut) | 
|  | { | 
|  | IUnknown_AddRef(iface); | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | WARN("unsupported interface: %s\n", debugstr_guid(iid)); | 
|  | return E_NOINTERFACE; | 
|  | } | 
|  |  | 
|  | static ULONG WINAPI DocumentMgr_AddRef(ITfDocumentMgr *iface) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfDocumentMgr(iface); | 
|  | return InterlockedIncrement(&This->refCount); | 
|  | } | 
|  |  | 
|  | static ULONG WINAPI DocumentMgr_Release(ITfDocumentMgr *iface) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfDocumentMgr(iface); | 
|  | ULONG ret; | 
|  |  | 
|  | ret = InterlockedDecrement(&This->refCount); | 
|  | if (ret == 0) | 
|  | DocumentMgr_Destructor(This); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | /***************************************************** | 
|  | * ITfDocumentMgr functions | 
|  | *****************************************************/ | 
|  | static HRESULT WINAPI DocumentMgr_CreateContext(ITfDocumentMgr *iface, | 
|  | TfClientId tidOwner, | 
|  | DWORD dwFlags, IUnknown *punk, ITfContext **ppic, | 
|  | TfEditCookie *pecTextStore) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfDocumentMgr(iface); | 
|  | TRACE("(%p) 0x%x 0x%x %p %p %p\n",This,tidOwner,dwFlags,punk,ppic,pecTextStore); | 
|  | return Context_Constructor(tidOwner, punk, iface, ppic, pecTextStore); | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI DocumentMgr_Push(ITfDocumentMgr *iface, ITfContext *pic) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfDocumentMgr(iface); | 
|  | ITfContext *check; | 
|  |  | 
|  | TRACE("(%p) %p\n",This,pic); | 
|  |  | 
|  | if (This->contextStack[1])  /* FUll */ | 
|  | return TF_E_STACKFULL; | 
|  |  | 
|  | if (!pic || FAILED(IUnknown_QueryInterface(pic,&IID_ITfContext,(LPVOID*) &check))) | 
|  | return E_INVALIDARG; | 
|  |  | 
|  | if (This->contextStack[0] == NULL) | 
|  | ITfThreadMgrEventSink_OnInitDocumentMgr(This->ThreadMgrSink,iface); | 
|  |  | 
|  | This->contextStack[1] = This->contextStack[0]; | 
|  | This->contextStack[0] = check; | 
|  |  | 
|  | Context_Initialize(check, iface); | 
|  | ITfThreadMgrEventSink_OnPushContext(This->ThreadMgrSink,check); | 
|  |  | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI DocumentMgr_Pop(ITfDocumentMgr *iface, DWORD dwFlags) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfDocumentMgr(iface); | 
|  | TRACE("(%p) 0x%x\n",This,dwFlags); | 
|  |  | 
|  | if (dwFlags == TF_POPF_ALL) | 
|  | { | 
|  | if (This->contextStack[0]) | 
|  | { | 
|  | ITfThreadMgrEventSink_OnPopContext(This->ThreadMgrSink,This->contextStack[0]); | 
|  | ITfContext_Release(This->contextStack[0]); | 
|  | Context_Uninitialize(This->contextStack[0]); | 
|  | } | 
|  | if (This->contextStack[1]) | 
|  | { | 
|  | ITfThreadMgrEventSink_OnPopContext(This->ThreadMgrSink,This->contextStack[1]); | 
|  | ITfContext_Release(This->contextStack[1]); | 
|  | Context_Uninitialize(This->contextStack[1]); | 
|  | } | 
|  | This->contextStack[0] = This->contextStack[1] = NULL; | 
|  | ITfThreadMgrEventSink_OnUninitDocumentMgr(This->ThreadMgrSink, iface); | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | if (dwFlags) | 
|  | return E_INVALIDARG; | 
|  |  | 
|  | if (This->contextStack[1] == NULL) /* Cannot pop last context */ | 
|  | return E_FAIL; | 
|  |  | 
|  | ITfThreadMgrEventSink_OnPopContext(This->ThreadMgrSink,This->contextStack[0]); | 
|  | ITfContext_Release(This->contextStack[0]); | 
|  | Context_Uninitialize(This->contextStack[0]); | 
|  | This->contextStack[0] = This->contextStack[1]; | 
|  | This->contextStack[1] = NULL; | 
|  |  | 
|  | if (This->contextStack[0] == NULL) | 
|  | ITfThreadMgrEventSink_OnUninitDocumentMgr(This->ThreadMgrSink, iface); | 
|  |  | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI DocumentMgr_GetTop(ITfDocumentMgr *iface, ITfContext **ppic) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfDocumentMgr(iface); | 
|  | TRACE("(%p)\n",This); | 
|  | if (!ppic) | 
|  | return E_INVALIDARG; | 
|  |  | 
|  | if (This->contextStack[0]) | 
|  | ITfContext_AddRef(This->contextStack[0]); | 
|  |  | 
|  | *ppic = This->contextStack[0]; | 
|  |  | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI DocumentMgr_GetBase(ITfDocumentMgr *iface, ITfContext **ppic) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfDocumentMgr(iface); | 
|  | ITfContext *tgt; | 
|  |  | 
|  | TRACE("(%p)\n",This); | 
|  | if (!ppic) | 
|  | return E_INVALIDARG; | 
|  |  | 
|  | if (This->contextStack[1]) | 
|  | tgt = This->contextStack[1]; | 
|  | else | 
|  | tgt = This->contextStack[0]; | 
|  |  | 
|  | if (tgt) | 
|  | ITfContext_AddRef(tgt); | 
|  |  | 
|  | *ppic = tgt; | 
|  |  | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI DocumentMgr_EnumContexts(ITfDocumentMgr *iface, IEnumTfContexts **ppEnum) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfDocumentMgr(iface); | 
|  | TRACE("(%p) %p\n",This,ppEnum); | 
|  | return EnumTfContext_Constructor(This, ppEnum); | 
|  | } | 
|  |  | 
|  | static const ITfDocumentMgrVtbl DocumentMgr_DocumentMgrVtbl = | 
|  | { | 
|  | DocumentMgr_QueryInterface, | 
|  | DocumentMgr_AddRef, | 
|  | DocumentMgr_Release, | 
|  |  | 
|  | DocumentMgr_CreateContext, | 
|  | DocumentMgr_Push, | 
|  | DocumentMgr_Pop, | 
|  | DocumentMgr_GetTop, | 
|  | DocumentMgr_GetBase, | 
|  | DocumentMgr_EnumContexts | 
|  | }; | 
|  |  | 
|  |  | 
|  | static HRESULT WINAPI Source_QueryInterface(ITfSource *iface, REFIID iid, LPVOID *ppvOut) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfSource(iface); | 
|  | return DocumentMgr_QueryInterface(&This->ITfDocumentMgr_iface, iid, *ppvOut); | 
|  | } | 
|  |  | 
|  | static ULONG WINAPI Source_AddRef(ITfSource *iface) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfSource(iface); | 
|  | return DocumentMgr_AddRef(&This->ITfDocumentMgr_iface); | 
|  | } | 
|  |  | 
|  | static ULONG WINAPI Source_Release(ITfSource *iface) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfSource(iface); | 
|  | return DocumentMgr_Release(&This->ITfDocumentMgr_iface); | 
|  | } | 
|  |  | 
|  | /***************************************************** | 
|  | * ITfSource functions | 
|  | *****************************************************/ | 
|  | static HRESULT WINAPI DocumentMgrSource_AdviseSink(ITfSource *iface, | 
|  | REFIID riid, IUnknown *punk, DWORD *pdwCookie) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfSource(iface); | 
|  | FIXME("STUB:(%p)\n",This); | 
|  | return E_NOTIMPL; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI DocumentMgrSource_UnadviseSink(ITfSource *iface, DWORD pdwCookie) | 
|  | { | 
|  | DocumentMgr *This = impl_from_ITfSource(iface); | 
|  | FIXME("STUB:(%p)\n",This); | 
|  | return E_NOTIMPL; | 
|  | } | 
|  |  | 
|  | static const ITfSourceVtbl DocumentMgr_SourceVtbl = | 
|  | { | 
|  | Source_QueryInterface, | 
|  | Source_AddRef, | 
|  | Source_Release, | 
|  |  | 
|  | DocumentMgrSource_AdviseSink, | 
|  | DocumentMgrSource_UnadviseSink, | 
|  | }; | 
|  |  | 
|  | HRESULT DocumentMgr_Constructor(ITfThreadMgrEventSink *ThreadMgrSink, ITfDocumentMgr **ppOut) | 
|  | { | 
|  | DocumentMgr *This; | 
|  |  | 
|  | This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(DocumentMgr)); | 
|  | if (This == NULL) | 
|  | return E_OUTOFMEMORY; | 
|  |  | 
|  | This->ITfDocumentMgr_iface.lpVtbl = &DocumentMgr_DocumentMgrVtbl; | 
|  | This->ITfSource_iface.lpVtbl = &DocumentMgr_SourceVtbl; | 
|  | This->refCount = 1; | 
|  | This->ThreadMgrSink = ThreadMgrSink; | 
|  |  | 
|  | CompartmentMgr_Constructor((IUnknown*)This, &IID_IUnknown, (IUnknown**)&This->CompartmentMgr); | 
|  |  | 
|  | TRACE("returning %p\n", This); | 
|  | *ppOut = &This->ITfDocumentMgr_iface; | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | /************************************************** | 
|  | * IEnumTfContexts implementaion | 
|  | **************************************************/ | 
|  | static void EnumTfContext_Destructor(EnumTfContext *This) | 
|  | { | 
|  | TRACE("destroying %p\n", This); | 
|  | HeapFree(GetProcessHeap(),0,This); | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI EnumTfContext_QueryInterface(IEnumTfContexts *iface, REFIID iid, LPVOID *ppvOut) | 
|  | { | 
|  | EnumTfContext *This = impl_from_IEnumTfContexts(iface); | 
|  | *ppvOut = NULL; | 
|  |  | 
|  | if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_IEnumTfContexts)) | 
|  | { | 
|  | *ppvOut = This; | 
|  | } | 
|  |  | 
|  | if (*ppvOut) | 
|  | { | 
|  | IUnknown_AddRef(iface); | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | WARN("unsupported interface: %s\n", debugstr_guid(iid)); | 
|  | return E_NOINTERFACE; | 
|  | } | 
|  |  | 
|  | static ULONG WINAPI EnumTfContext_AddRef(IEnumTfContexts *iface) | 
|  | { | 
|  | EnumTfContext *This = impl_from_IEnumTfContexts(iface); | 
|  | return InterlockedIncrement(&This->refCount); | 
|  | } | 
|  |  | 
|  | static ULONG WINAPI EnumTfContext_Release(IEnumTfContexts *iface) | 
|  | { | 
|  | EnumTfContext *This = impl_from_IEnumTfContexts(iface); | 
|  | ULONG ret; | 
|  |  | 
|  | ret = InterlockedDecrement(&This->refCount); | 
|  | if (ret == 0) | 
|  | EnumTfContext_Destructor(This); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI EnumTfContext_Next(IEnumTfContexts *iface, | 
|  | ULONG ulCount, ITfContext **rgContext, ULONG *pcFetched) | 
|  | { | 
|  | EnumTfContext *This = impl_from_IEnumTfContexts(iface); | 
|  | ULONG fetched = 0; | 
|  |  | 
|  | TRACE("(%p)\n",This); | 
|  |  | 
|  | if (rgContext == NULL) return E_POINTER; | 
|  |  | 
|  | while (fetched < ulCount) | 
|  | { | 
|  | if (This->index > 1) | 
|  | break; | 
|  |  | 
|  | if (!This->docmgr->contextStack[This->index]) | 
|  | break; | 
|  |  | 
|  | *rgContext = This->docmgr->contextStack[This->index]; | 
|  | ITfContext_AddRef(*rgContext); | 
|  |  | 
|  | ++This->index; | 
|  | ++fetched; | 
|  | ++rgContext; | 
|  | } | 
|  |  | 
|  | if (pcFetched) *pcFetched = fetched; | 
|  | return fetched == ulCount ? S_OK : S_FALSE; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI EnumTfContext_Skip( IEnumTfContexts* iface, ULONG celt) | 
|  | { | 
|  | EnumTfContext *This = impl_from_IEnumTfContexts(iface); | 
|  | TRACE("(%p)\n",This); | 
|  | This->index += celt; | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI EnumTfContext_Reset( IEnumTfContexts* iface) | 
|  | { | 
|  | EnumTfContext *This = impl_from_IEnumTfContexts(iface); | 
|  | TRACE("(%p)\n",This); | 
|  | This->index = 0; | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI EnumTfContext_Clone( IEnumTfContexts *iface, | 
|  | IEnumTfContexts **ppenum) | 
|  | { | 
|  | EnumTfContext *This = impl_from_IEnumTfContexts(iface); | 
|  | HRESULT res; | 
|  |  | 
|  | TRACE("(%p)\n",This); | 
|  |  | 
|  | if (ppenum == NULL) return E_POINTER; | 
|  |  | 
|  | res = EnumTfContext_Constructor(This->docmgr, ppenum); | 
|  | if (SUCCEEDED(res)) | 
|  | { | 
|  | EnumTfContext *new_This = impl_from_IEnumTfContexts(*ppenum); | 
|  | new_This->index = This->index; | 
|  | } | 
|  | return res; | 
|  | } | 
|  |  | 
|  | static const IEnumTfContextsVtbl IEnumTfContexts_Vtbl ={ | 
|  | EnumTfContext_QueryInterface, | 
|  | EnumTfContext_AddRef, | 
|  | EnumTfContext_Release, | 
|  |  | 
|  | EnumTfContext_Clone, | 
|  | EnumTfContext_Next, | 
|  | EnumTfContext_Reset, | 
|  | EnumTfContext_Skip | 
|  | }; | 
|  |  | 
|  | static HRESULT EnumTfContext_Constructor(DocumentMgr *mgr, IEnumTfContexts **ppOut) | 
|  | { | 
|  | EnumTfContext *This; | 
|  |  | 
|  | This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(EnumTfContext)); | 
|  | if (This == NULL) | 
|  | return E_OUTOFMEMORY; | 
|  |  | 
|  | This->IEnumTfContexts_iface.lpVtbl = &IEnumTfContexts_Vtbl; | 
|  | This->refCount = 1; | 
|  | This->docmgr = mgr; | 
|  |  | 
|  | TRACE("returning %p\n", This); | 
|  | *ppOut = &This->IEnumTfContexts_iface; | 
|  | return S_OK; | 
|  | } |