/* * ITfContext 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 #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 "olectl.h" #include "wine/unicode.h" #include "msctf.h" #include "msctf_internal.h" WINE_DEFAULT_DEBUG_CHANNEL(msctf); typedef struct tagContext { ITfContext ITfContext_iface; ITfSource ITfSource_iface; /* const ITfContextCompositionVtbl *ContextCompositionVtbl; */ /* const ITfContextOwnerCompositionServicesVtbl *ContextOwnerCompositionServicesVtbl; */ /* const ITfContextOwnerServicesVtbl *ContextOwnerServicesVtbl; */ ITfInsertAtSelection ITfInsertAtSelection_iface; /* const ITfMouseTrackerVtbl *MouseTrackerVtbl; */ /* const ITfQueryEmbeddedVtbl *QueryEmbeddedVtbl; */ ITfSourceSingle ITfSourceSingle_iface; ITextStoreACPSink ITextStoreACPSink_iface; ITextStoreACPServices ITextStoreACPServices_iface; LONG refCount; BOOL connected; /* Aggregation */ ITfCompartmentMgr *CompartmentMgr; TfClientId tidOwner; TfEditCookie defaultCookie; TS_STATUS documentStatus; ITfDocumentMgr *manager; ITextStoreACP *pITextStoreACP; ITfContextOwnerCompositionSink *pITfContextOwnerCompositionSink; ITfEditSession* currentEditSession; /* kept as separate lists to reduce unnecessary iterations */ struct list pContextKeyEventSink; struct list pEditTransactionSink; struct list pStatusSink; struct list pTextEditSink; struct list pTextLayoutSink; } Context; typedef struct tagEditCookie { DWORD lockType; Context *pOwningContext; } EditCookie; static inline Context *impl_from_ITfContext(ITfContext *iface) { return CONTAINING_RECORD(iface, Context, ITfContext_iface); } static inline Context *impl_from_ITfSource(ITfSource *iface) { return CONTAINING_RECORD(iface, Context, ITfSource_iface); } static inline Context *impl_from_ITfInsertAtSelection(ITfInsertAtSelection *iface) { return CONTAINING_RECORD(iface, Context, ITfInsertAtSelection_iface); } static inline Context *impl_from_ITfSourceSingle(ITfSourceSingle* iface) { return CONTAINING_RECORD(iface, Context, ITfSourceSingle_iface); } static inline Context *impl_from_ITextStoreACPSink(ITextStoreACPSink *iface) { return CONTAINING_RECORD(iface, Context, ITextStoreACPSink_iface); } static inline Context *impl_from_ITextStoreACPServices(ITextStoreACPServices *iface) { return CONTAINING_RECORD(iface, Context, ITextStoreACPServices_iface); } static void Context_Destructor(Context *This) { EditCookie *cookie; TRACE("destroying %p\n", This); if (This->pITextStoreACP) ITextStoreACP_Release(This->pITextStoreACP); if (This->pITfContextOwnerCompositionSink) ITfContextOwnerCompositionSink_Release(This->pITfContextOwnerCompositionSink); if (This->defaultCookie) { cookie = remove_Cookie(This->defaultCookie); HeapFree(GetProcessHeap(),0,cookie); This->defaultCookie = 0; } free_sinks(&This->pContextKeyEventSink); free_sinks(&This->pEditTransactionSink); free_sinks(&This->pStatusSink); free_sinks(&This->pTextEditSink); free_sinks(&This->pTextLayoutSink); CompartmentMgr_Destructor(This->CompartmentMgr); HeapFree(GetProcessHeap(),0,This); } static HRESULT WINAPI Context_QueryInterface(ITfContext *iface, REFIID iid, LPVOID *ppvOut) { Context *This = impl_from_ITfContext(iface); *ppvOut = NULL; if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_ITfContext)) { *ppvOut = &This->ITfContext_iface; } else if (IsEqualIID(iid, &IID_ITfSource)) { *ppvOut = &This->ITfSource_iface; } else if (IsEqualIID(iid, &IID_ITfInsertAtSelection)) { *ppvOut = &This->ITfInsertAtSelection_iface; } else if (IsEqualIID(iid, &IID_ITfCompartmentMgr)) { *ppvOut = This->CompartmentMgr; } else if (IsEqualIID(iid, &IID_ITfSourceSingle)) { *ppvOut = &This->ITfSourceSingle_iface; } if (*ppvOut) { ITfContext_AddRef(iface); return S_OK; } WARN("unsupported interface: %s\n", debugstr_guid(iid)); return E_NOINTERFACE; } static ULONG WINAPI Context_AddRef(ITfContext *iface) { Context *This = impl_from_ITfContext(iface); return InterlockedIncrement(&This->refCount); } static ULONG WINAPI Context_Release(ITfContext *iface) { Context *This = impl_from_ITfContext(iface); ULONG ret; ret = InterlockedDecrement(&This->refCount); if (ret == 0) Context_Destructor(This); return ret; } /***************************************************** * ITfContext functions *****************************************************/ static HRESULT WINAPI Context_RequestEditSession (ITfContext *iface, TfClientId tid, ITfEditSession *pes, DWORD dwFlags, HRESULT *phrSession) { Context *This = impl_from_ITfContext(iface); HRESULT hr; DWORD dwLockFlags = 0x0; TRACE("(%p) %i %p %x %p\n",This, tid, pes, dwFlags, phrSession); if (!(dwFlags & TF_ES_READ) && !(dwFlags & TF_ES_READWRITE)) { *phrSession = E_FAIL; return E_INVALIDARG; } if (!This->pITextStoreACP) { FIXME("No ITextStoreACP available\n"); *phrSession = E_FAIL; return E_FAIL; } if (!(dwFlags & TF_ES_ASYNC)) dwLockFlags |= TS_LF_SYNC; if ((dwFlags & TF_ES_READWRITE) == TF_ES_READWRITE) dwLockFlags |= TS_LF_READWRITE; else if (dwFlags & TF_ES_READ) dwLockFlags |= TS_LF_READ; if (!This->documentStatus.dwDynamicFlags) ITextStoreACP_GetStatus(This->pITextStoreACP, &This->documentStatus); if (((dwFlags & TF_ES_READWRITE) == TF_ES_READWRITE) && (This->documentStatus.dwDynamicFlags & TS_SD_READONLY)) { *phrSession = TS_E_READONLY; return S_OK; } if (FAILED (ITfEditSession_QueryInterface(pes, &IID_ITfEditSession, (LPVOID*)&This->currentEditSession))) { *phrSession = E_FAIL; return E_INVALIDARG; } hr = ITextStoreACP_RequestLock(This->pITextStoreACP, dwLockFlags, phrSession); return hr; } static HRESULT WINAPI Context_InWriteSession (ITfContext *iface, TfClientId tid, BOOL *pfWriteSession) { Context *This = impl_from_ITfContext(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static HRESULT WINAPI Context_GetSelection (ITfContext *iface, TfEditCookie ec, ULONG ulIndex, ULONG ulCount, TF_SELECTION *pSelection, ULONG *pcFetched) { Context *This = impl_from_ITfContext(iface); EditCookie *cookie; ULONG count, i; ULONG totalFetched = 0; HRESULT hr = S_OK; if (!pSelection || !pcFetched) return E_INVALIDARG; *pcFetched = 0; if (!This->connected) return TF_E_DISCONNECTED; if (get_Cookie_magic(ec)!=COOKIE_MAGIC_EDITCOOKIE) return TF_E_NOLOCK; if (!This->pITextStoreACP) { FIXME("Context does not have a ITextStoreACP\n"); return E_NOTIMPL; } cookie = get_Cookie_data(ec); if (ulIndex == TF_DEFAULT_SELECTION) count = 1; else count = ulCount; for (i = 0; i < count; i++) { DWORD fetched; TS_SELECTION_ACP acps; hr = ITextStoreACP_GetSelection(This->pITextStoreACP, ulIndex + i, 1, &acps, &fetched); if (hr == TS_E_NOLOCK) return TF_E_NOLOCK; else if (SUCCEEDED(hr)) { pSelection[totalFetched].style.ase = acps.style.ase; pSelection[totalFetched].style.fInterimChar = acps.style.fInterimChar; Range_Constructor(iface, This->pITextStoreACP, cookie->lockType, acps.acpStart, acps.acpEnd, &pSelection[totalFetched].range); totalFetched ++; } else break; } *pcFetched = totalFetched; return hr; } static HRESULT WINAPI Context_SetSelection (ITfContext *iface, TfEditCookie ec, ULONG ulCount, const TF_SELECTION *pSelection) { Context *This = impl_from_ITfContext(iface); TS_SELECTION_ACP *acp; ULONG i; HRESULT hr; TRACE("(%p) %i %i %p\n",This,ec,ulCount,pSelection); if (!This->pITextStoreACP) { FIXME("Context does not have a ITextStoreACP\n"); return E_NOTIMPL; } if (get_Cookie_magic(ec)!=COOKIE_MAGIC_EDITCOOKIE) return TF_E_NOLOCK; acp = HeapAlloc(GetProcessHeap(), 0, sizeof(TS_SELECTION_ACP) * ulCount); if (!acp) return E_OUTOFMEMORY; for (i = 0; i < ulCount; i++) if (FAILED(TF_SELECTION_to_TS_SELECTION_ACP(&pSelection[i], &acp[i]))) { TRACE("Selection Conversion Failed\n"); HeapFree(GetProcessHeap(), 0 , acp); return E_FAIL; } hr = ITextStoreACP_SetSelection(This->pITextStoreACP, ulCount, acp); HeapFree(GetProcessHeap(), 0, acp); return hr; } static HRESULT WINAPI Context_GetStart (ITfContext *iface, TfEditCookie ec, ITfRange **ppStart) { Context *This = impl_from_ITfContext(iface); EditCookie *cookie; TRACE("(%p) %i %p\n",This,ec,ppStart); if (!ppStart) return E_INVALIDARG; *ppStart = NULL; if (!This->connected) return TF_E_DISCONNECTED; if (get_Cookie_magic(ec)!=COOKIE_MAGIC_EDITCOOKIE) return TF_E_NOLOCK; cookie = get_Cookie_data(ec); return Range_Constructor(iface, This->pITextStoreACP, cookie->lockType, 0, 0, ppStart); } static HRESULT WINAPI Context_GetEnd (ITfContext *iface, TfEditCookie ec, ITfRange **ppEnd) { Context *This = impl_from_ITfContext(iface); EditCookie *cookie; LONG end; TRACE("(%p) %i %p\n",This,ec,ppEnd); if (!ppEnd) return E_INVALIDARG; *ppEnd = NULL; if (!This->connected) return TF_E_DISCONNECTED; if (get_Cookie_magic(ec)!=COOKIE_MAGIC_EDITCOOKIE) return TF_E_NOLOCK; if (!This->pITextStoreACP) { FIXME("Context does not have a ITextStoreACP\n"); return E_NOTIMPL; } cookie = get_Cookie_data(ec); ITextStoreACP_GetEndACP(This->pITextStoreACP,&end); return Range_Constructor(iface, This->pITextStoreACP, cookie->lockType, end, end, ppEnd); } static HRESULT WINAPI Context_GetActiveView (ITfContext *iface, ITfContextView **ppView) { Context *This = impl_from_ITfContext(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static HRESULT WINAPI Context_EnumViews (ITfContext *iface, IEnumTfContextViews **ppEnum) { Context *This = impl_from_ITfContext(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static HRESULT WINAPI Context_GetStatus (ITfContext *iface, TF_STATUS *pdcs) { Context *This = impl_from_ITfContext(iface); TRACE("(%p) %p\n",This,pdcs); if (!This->connected) return TF_E_DISCONNECTED; if (!pdcs) return E_INVALIDARG; if (!This->pITextStoreACP) { FIXME("Context does not have a ITextStoreACP\n"); return E_NOTIMPL; } ITextStoreACP_GetStatus(This->pITextStoreACP, &This->documentStatus); *pdcs = This->documentStatus; return S_OK; } static HRESULT WINAPI Context_GetProperty (ITfContext *iface, REFGUID guidProp, ITfProperty **ppProp) { Context *This = impl_from_ITfContext(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static HRESULT WINAPI Context_GetAppProperty (ITfContext *iface, REFGUID guidProp, ITfReadOnlyProperty **ppProp) { Context *This = impl_from_ITfContext(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static HRESULT WINAPI Context_TrackProperties (ITfContext *iface, const GUID **prgProp, ULONG cProp, const GUID **prgAppProp, ULONG cAppProp, ITfReadOnlyProperty **ppProperty) { Context *This = impl_from_ITfContext(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static HRESULT WINAPI Context_EnumProperties (ITfContext *iface, IEnumTfProperties **ppEnum) { Context *This = impl_from_ITfContext(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static HRESULT WINAPI Context_GetDocumentMgr (ITfContext *iface, ITfDocumentMgr **ppDm) { Context *This = impl_from_ITfContext(iface); TRACE("(%p) %p\n",This,ppDm); if (!ppDm) return E_INVALIDARG; *ppDm = This->manager; if (!This->manager) return S_FALSE; ITfDocumentMgr_AddRef(This->manager); return S_OK; } static HRESULT WINAPI Context_CreateRangeBackup (ITfContext *iface, TfEditCookie ec, ITfRange *pRange, ITfRangeBackup **ppBackup) { Context *This = impl_from_ITfContext(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static const ITfContextVtbl ContextVtbl = { Context_QueryInterface, Context_AddRef, Context_Release, Context_RequestEditSession, Context_InWriteSession, Context_GetSelection, Context_SetSelection, Context_GetStart, Context_GetEnd, Context_GetActiveView, Context_EnumViews, Context_GetStatus, Context_GetProperty, Context_GetAppProperty, Context_TrackProperties, Context_EnumProperties, Context_GetDocumentMgr, Context_CreateRangeBackup }; static HRESULT WINAPI ContextSource_QueryInterface(ITfSource *iface, REFIID iid, LPVOID *ppvOut) { Context *This = impl_from_ITfSource(iface); return ITfContext_QueryInterface(&This->ITfContext_iface, iid, ppvOut); } static ULONG WINAPI ContextSource_AddRef(ITfSource *iface) { Context *This = impl_from_ITfSource(iface); return ITfContext_AddRef(&This->ITfContext_iface); } static ULONG WINAPI ContextSource_Release(ITfSource *iface) { Context *This = impl_from_ITfSource(iface); return ITfContext_Release(&This->ITfContext_iface); } /***************************************************** * ITfSource functions *****************************************************/ static HRESULT WINAPI ContextSource_AdviseSink(ITfSource *iface, REFIID riid, IUnknown *punk, DWORD *pdwCookie) { Context *This = impl_from_ITfSource(iface); TRACE("(%p) %s %p %p\n",This,debugstr_guid(riid),punk,pdwCookie); if (!riid || !punk || !pdwCookie) return E_INVALIDARG; if (IsEqualIID(riid, &IID_ITfTextEditSink)) return advise_sink(&This->pTextEditSink, &IID_ITfTextEditSink, COOKIE_MAGIC_CONTEXTSINK, punk, pdwCookie); FIXME("(%p) Unhandled Sink: %s\n",This,debugstr_guid(riid)); return E_NOTIMPL; } static HRESULT WINAPI ContextSource_UnadviseSink(ITfSource *iface, DWORD pdwCookie) { Context *This = impl_from_ITfSource(iface); TRACE("(%p) %x\n",This,pdwCookie); if (get_Cookie_magic(pdwCookie)!=COOKIE_MAGIC_CONTEXTSINK) return E_INVALIDARG; return unadvise_sink(pdwCookie); } static const ITfSourceVtbl ContextSourceVtbl = { ContextSource_QueryInterface, ContextSource_AddRef, ContextSource_Release, ContextSource_AdviseSink, ContextSource_UnadviseSink }; /***************************************************** * ITfInsertAtSelection functions *****************************************************/ static HRESULT WINAPI InsertAtSelection_QueryInterface(ITfInsertAtSelection *iface, REFIID iid, LPVOID *ppvOut) { Context *This = impl_from_ITfInsertAtSelection(iface); return ITfContext_QueryInterface(&This->ITfContext_iface, iid, ppvOut); } static ULONG WINAPI InsertAtSelection_AddRef(ITfInsertAtSelection *iface) { Context *This = impl_from_ITfInsertAtSelection(iface); return ITfContext_AddRef(&This->ITfContext_iface); } static ULONG WINAPI InsertAtSelection_Release(ITfInsertAtSelection *iface) { Context *This = impl_from_ITfInsertAtSelection(iface); return ITfContext_Release(&This->ITfContext_iface); } static HRESULT WINAPI InsertAtSelection_InsertTextAtSelection( ITfInsertAtSelection *iface, TfEditCookie ec, DWORD dwFlags, const WCHAR *pchText, LONG cch, ITfRange **ppRange) { Context *This = impl_from_ITfInsertAtSelection(iface); EditCookie *cookie; LONG acpStart, acpEnd; TS_TEXTCHANGE change; HRESULT hr; TRACE("(%p) %i %x %s %p\n",This, ec, dwFlags, debugstr_wn(pchText,cch), ppRange); if (!This->connected) return TF_E_DISCONNECTED; if (get_Cookie_magic(ec)!=COOKIE_MAGIC_EDITCOOKIE) return TF_E_NOLOCK; cookie = get_Cookie_data(ec); if ((cookie->lockType & TS_LF_READWRITE) != TS_LF_READWRITE ) return TS_E_READONLY; if (!This->pITextStoreACP) { FIXME("Context does not have a ITextStoreACP\n"); return E_NOTIMPL; } hr = ITextStoreACP_InsertTextAtSelection(This->pITextStoreACP, dwFlags, pchText, cch, &acpStart, &acpEnd, &change); if (SUCCEEDED(hr)) Range_Constructor(&This->ITfContext_iface, This->pITextStoreACP, cookie->lockType, change.acpStart, change.acpNewEnd, ppRange); return hr; } static HRESULT WINAPI InsertAtSelection_InsertEmbeddedAtSelection( ITfInsertAtSelection *iface, TfEditCookie ec, DWORD dwFlags, IDataObject *pDataObject, ITfRange **ppRange) { Context *This = impl_from_ITfInsertAtSelection(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static const ITfInsertAtSelectionVtbl InsertAtSelectionVtbl = { InsertAtSelection_QueryInterface, InsertAtSelection_AddRef, InsertAtSelection_Release, InsertAtSelection_InsertTextAtSelection, InsertAtSelection_InsertEmbeddedAtSelection, }; /***************************************************** * ITfSourceSingle functions *****************************************************/ static HRESULT WINAPI SourceSingle_QueryInterface(ITfSourceSingle *iface, REFIID iid, LPVOID *ppvOut) { Context *This = impl_from_ITfSourceSingle(iface); return ITfContext_QueryInterface(&This->ITfContext_iface, iid, ppvOut); } static ULONG WINAPI SourceSingle_AddRef(ITfSourceSingle *iface) { Context *This = impl_from_ITfSourceSingle(iface); return ITfContext_AddRef(&This->ITfContext_iface); } static ULONG WINAPI SourceSingle_Release(ITfSourceSingle *iface) { Context *This = impl_from_ITfSourceSingle(iface); return ITfContext_Release(&This->ITfContext_iface); } static HRESULT WINAPI SourceSingle_AdviseSingleSink( ITfSourceSingle *iface, TfClientId tid, REFIID riid, IUnknown *punk) { Context *This = impl_from_ITfSourceSingle(iface); FIXME("STUB:(%p) %i %s %p\n",This, tid, debugstr_guid(riid),punk); return E_NOTIMPL; } static HRESULT WINAPI SourceSingle_UnadviseSingleSink( ITfSourceSingle *iface, TfClientId tid, REFIID riid) { Context *This = impl_from_ITfSourceSingle(iface); FIXME("STUB:(%p) %i %s\n",This, tid, debugstr_guid(riid)); return E_NOTIMPL; } static const ITfSourceSingleVtbl ContextSourceSingleVtbl = { SourceSingle_QueryInterface, SourceSingle_AddRef, SourceSingle_Release, SourceSingle_AdviseSingleSink, SourceSingle_UnadviseSingleSink, }; /************************************************************************** * ITextStoreACPSink **************************************************************************/ static HRESULT WINAPI TextStoreACPSink_QueryInterface(ITextStoreACPSink *iface, REFIID iid, LPVOID *ppvOut) { Context *This = impl_from_ITextStoreACPSink(iface); *ppvOut = NULL; if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_ITextStoreACPSink)) { *ppvOut = &This->ITextStoreACPSink_iface; } else if (IsEqualIID(iid, &IID_ITextStoreACPServices)) *ppvOut = &This->ITextStoreACPServices_iface; if (*ppvOut) { ITextStoreACPSink_AddRef(iface); return S_OK; } WARN("unsupported interface: %s\n", debugstr_guid(iid)); return E_NOINTERFACE; } static ULONG WINAPI TextStoreACPSink_AddRef(ITextStoreACPSink *iface) { Context *This = impl_from_ITextStoreACPSink(iface); return ITfContext_AddRef(&This->ITfContext_iface); } static ULONG WINAPI TextStoreACPSink_Release(ITextStoreACPSink *iface) { Context *This = impl_from_ITextStoreACPSink(iface); return ITfContext_Release(&This->ITfContext_iface); } /***************************************************** * ITextStoreACPSink functions *****************************************************/ static HRESULT WINAPI TextStoreACPSink_OnTextChange(ITextStoreACPSink *iface, DWORD dwFlags, const TS_TEXTCHANGE *pChange) { Context *This = impl_from_ITextStoreACPSink(iface); FIXME("STUB:(%p)\n",This); return S_OK; } static HRESULT WINAPI TextStoreACPSink_OnSelectionChange(ITextStoreACPSink *iface) { Context *This = impl_from_ITextStoreACPSink(iface); FIXME("STUB:(%p)\n",This); return S_OK; } static HRESULT WINAPI TextStoreACPSink_OnLayoutChange(ITextStoreACPSink *iface, TsLayoutCode lcode, TsViewCookie vcView) { Context *This = impl_from_ITextStoreACPSink(iface); FIXME("STUB:(%p)\n",This); return S_OK; } static HRESULT WINAPI TextStoreACPSink_OnStatusChange(ITextStoreACPSink *iface, DWORD dwFlags) { Context *This = impl_from_ITextStoreACPSink(iface); HRESULT hr, hrSession; TRACE("(%p) %x\n",This, dwFlags); if (!This->pITextStoreACP) { FIXME("Context does not have a ITextStoreACP\n"); return E_NOTIMPL; } hr = ITextStoreACP_RequestLock(This->pITextStoreACP, TS_LF_READ, &hrSession); if(SUCCEEDED(hr) && SUCCEEDED(hrSession)) This->documentStatus.dwDynamicFlags = dwFlags; return S_OK; } static HRESULT WINAPI TextStoreACPSink_OnAttrsChange(ITextStoreACPSink *iface, LONG acpStart, LONG acpEnd, ULONG cAttrs, const TS_ATTRID *paAttrs) { Context *This = impl_from_ITextStoreACPSink(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static HRESULT WINAPI TextStoreACPSink_OnLockGranted(ITextStoreACPSink *iface, DWORD dwLockFlags) { Context *This = impl_from_ITextStoreACPSink(iface); HRESULT hr; EditCookie *cookie,*sinkcookie; TfEditCookie ec; struct list *cursor; TRACE("(%p) %x\n",This, dwLockFlags); if (!This->currentEditSession) { FIXME("OnLockGranted called for something other than an EditSession\n"); return S_OK; } cookie = HeapAlloc(GetProcessHeap(),0,sizeof(EditCookie)); if (!cookie) return E_OUTOFMEMORY; sinkcookie = HeapAlloc(GetProcessHeap(),0,sizeof(EditCookie)); if (!sinkcookie) { HeapFree(GetProcessHeap(), 0, cookie); return E_OUTOFMEMORY; } cookie->lockType = dwLockFlags; cookie->pOwningContext = This; ec = generate_Cookie(COOKIE_MAGIC_EDITCOOKIE, cookie); hr = ITfEditSession_DoEditSession(This->currentEditSession, ec); if ((dwLockFlags&TS_LF_READWRITE) == TS_LF_READWRITE) { ITfTextEditSink *sink; TfEditCookie sc; sinkcookie->lockType = TS_LF_READ; sinkcookie->pOwningContext = This; sc = generate_Cookie(COOKIE_MAGIC_EDITCOOKIE, sinkcookie); /*TODO: implement ITfEditRecord */ SINK_FOR_EACH(cursor, &This->pTextEditSink, ITfTextEditSink, sink) { ITfTextEditSink_OnEndEdit(sink, (ITfContext*) &This->ITfContext_iface, sc, NULL); } sinkcookie = remove_Cookie(sc); } HeapFree(GetProcessHeap(),0,sinkcookie); ITfEditSession_Release(This->currentEditSession); This->currentEditSession = NULL; /* Edit Cookie is only valid during the edit session */ cookie = remove_Cookie(ec); HeapFree(GetProcessHeap(),0,cookie); return hr; } static HRESULT WINAPI TextStoreACPSink_OnStartEditTransaction(ITextStoreACPSink *iface) { Context *This = impl_from_ITextStoreACPSink(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static HRESULT WINAPI TextStoreACPSink_OnEndEditTransaction(ITextStoreACPSink *iface) { Context *This = impl_from_ITextStoreACPSink(iface); FIXME("STUB:(%p)\n",This); return E_NOTIMPL; } static const ITextStoreACPSinkVtbl TextStoreACPSinkVtbl = { TextStoreACPSink_QueryInterface, TextStoreACPSink_AddRef, TextStoreACPSink_Release, TextStoreACPSink_OnTextChange, TextStoreACPSink_OnSelectionChange, TextStoreACPSink_OnLayoutChange, TextStoreACPSink_OnStatusChange, TextStoreACPSink_OnAttrsChange, TextStoreACPSink_OnLockGranted, TextStoreACPSink_OnStartEditTransaction, TextStoreACPSink_OnEndEditTransaction }; static HRESULT WINAPI TextStoreACPServices_QueryInterface(ITextStoreACPServices *iface, REFIID riid, void **obj) { Context *This = impl_from_ITextStoreACPServices(iface); return ITextStoreACPSink_QueryInterface(&This->ITextStoreACPSink_iface, riid, obj); } static ULONG WINAPI TextStoreACPServices_AddRef(ITextStoreACPServices *iface) { Context *This = impl_from_ITextStoreACPServices(iface); return ITextStoreACPSink_AddRef(&This->ITextStoreACPSink_iface); } static ULONG WINAPI TextStoreACPServices_Release(ITextStoreACPServices *iface) { Context *This = impl_from_ITextStoreACPServices(iface); return ITextStoreACPSink_Release(&This->ITextStoreACPSink_iface); } static HRESULT WINAPI TextStoreACPServices_Serialize(ITextStoreACPServices *iface, ITfProperty *prop, ITfRange *range, TF_PERSISTENT_PROPERTY_HEADER_ACP *header, IStream *stream) { Context *This = impl_from_ITextStoreACPServices(iface); FIXME("stub: %p %p %p %p %p\n", This, prop, range, header, stream); return E_NOTIMPL; } static HRESULT WINAPI TextStoreACPServices_Unserialize(ITextStoreACPServices *iface, ITfProperty *prop, const TF_PERSISTENT_PROPERTY_HEADER_ACP *header, IStream *stream, ITfPersistentPropertyLoaderACP *loader) { Context *This = impl_from_ITextStoreACPServices(iface); FIXME("stub: %p %p %p %p %p\n", This, prop, header, stream, loader); return E_NOTIMPL; } static HRESULT WINAPI TextStoreACPServices_ForceLoadProperty(ITextStoreACPServices *iface, ITfProperty *prop) { Context *This = impl_from_ITextStoreACPServices(iface); FIXME("stub: %p %p\n", This, prop); return E_NOTIMPL; } static HRESULT WINAPI TextStoreACPServices_CreateRange(ITextStoreACPServices *iface, LONG start, LONG end, ITfRangeACP **range) { Context *This = impl_from_ITextStoreACPServices(iface); FIXME("stub: %p %d %d %p\n", This, start, end, range); return S_OK; } static const ITextStoreACPServicesVtbl TextStoreACPServicesVtbl = { TextStoreACPServices_QueryInterface, TextStoreACPServices_AddRef, TextStoreACPServices_Release, TextStoreACPServices_Serialize, TextStoreACPServices_Unserialize, TextStoreACPServices_ForceLoadProperty, TextStoreACPServices_CreateRange }; HRESULT Context_Constructor(TfClientId tidOwner, IUnknown *punk, ITfDocumentMgr *mgr, ITfContext **ppOut, TfEditCookie *pecTextStore) { Context *This; EditCookie *cookie; This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(Context)); if (This == NULL) return E_OUTOFMEMORY; cookie = HeapAlloc(GetProcessHeap(),0,sizeof(EditCookie)); if (cookie == NULL) { HeapFree(GetProcessHeap(),0,This); return E_OUTOFMEMORY; } TRACE("(%p) %x %p %p %p\n",This, tidOwner, punk, ppOut, pecTextStore); This->ITfContext_iface.lpVtbl= &ContextVtbl; This->ITfSource_iface.lpVtbl = &ContextSourceVtbl; This->ITfInsertAtSelection_iface.lpVtbl = &InsertAtSelectionVtbl; This->ITfSourceSingle_iface.lpVtbl = &ContextSourceSingleVtbl; This->ITextStoreACPSink_iface.lpVtbl = &TextStoreACPSinkVtbl; This->ITextStoreACPServices_iface.lpVtbl = &TextStoreACPServicesVtbl; This->refCount = 1; This->tidOwner = tidOwner; This->connected = FALSE; This->manager = mgr; CompartmentMgr_Constructor((IUnknown*)&This->ITfContext_iface, &IID_IUnknown, (IUnknown**)&This->CompartmentMgr); cookie->lockType = TF_ES_READ; cookie->pOwningContext = This; if (punk) { IUnknown_QueryInterface(punk, &IID_ITextStoreACP, (LPVOID*)&This->pITextStoreACP); IUnknown_QueryInterface(punk, &IID_ITfContextOwnerCompositionSink, (LPVOID*)&This->pITfContextOwnerCompositionSink); if (!This->pITextStoreACP && !This->pITfContextOwnerCompositionSink) FIXME("Unhandled pUnk\n"); } This->defaultCookie = generate_Cookie(COOKIE_MAGIC_EDITCOOKIE,cookie); *pecTextStore = This->defaultCookie; list_init(&This->pContextKeyEventSink); list_init(&This->pEditTransactionSink); list_init(&This->pStatusSink); list_init(&This->pTextEditSink); list_init(&This->pTextLayoutSink); *ppOut = &This->ITfContext_iface; TRACE("returning %p\n", *ppOut); return S_OK; } HRESULT Context_Initialize(ITfContext *iface, ITfDocumentMgr *manager) { Context *This = impl_from_ITfContext(iface); if (This->pITextStoreACP) ITextStoreACP_AdviseSink(This->pITextStoreACP, &IID_ITextStoreACPSink, (IUnknown*)&This->ITextStoreACPSink_iface, TS_AS_ALL_SINKS); This->connected = TRUE; This->manager = manager; return S_OK; } HRESULT Context_Uninitialize(ITfContext *iface) { Context *This = impl_from_ITfContext(iface); if (This->pITextStoreACP) ITextStoreACP_UnadviseSink(This->pITextStoreACP, (IUnknown*)&This->ITextStoreACPSink_iface); This->connected = FALSE; This->manager = NULL; return S_OK; }