| /* |
| * ITfThreadMgr implementation |
| * |
| * Copyright 2008 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 "olectl.h" |
| |
| #include "wine/unicode.h" |
| #include "wine/list.h" |
| |
| #include "msctf.h" |
| #include "msctf_internal.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(msctf); |
| |
| typedef struct tagThreadMgrSink { |
| struct list entry; |
| union { |
| /* ThreadMgr Sinks */ |
| IUnknown *pIUnknown; |
| /* ITfActiveLanguageProfileNotifySink *pITfActiveLanguageProfileNotifySink; */ |
| /* ITfDisplayAttributeNotifySink *pITfDisplayAttributeNotifySink; */ |
| /* ITfKeyTraceEventSink *pITfKeyTraceEventSink; */ |
| /* ITfPreservedKeyNotifySink *pITfPreservedKeyNotifySink; */ |
| /* ITfThreadFocusSink *pITfThreadFocusSink; */ |
| ITfThreadMgrEventSink *pITfThreadMgrEventSink; |
| } interfaces; |
| } ThreadMgrSink; |
| |
| typedef struct tagACLMulti { |
| const ITfThreadMgrVtbl *ThreadMgrVtbl; |
| const ITfSourceVtbl *SourceVtbl; |
| LONG refCount; |
| |
| const ITfThreadMgrEventSinkVtbl *ThreadMgrEventSinkVtbl; /* internal */ |
| |
| ITfDocumentMgr *focus; |
| |
| /* kept as separate lists to reduce unnecessary iterations */ |
| struct list ActiveLanguageProfileNotifySink; |
| struct list DisplayAttributeNotifySink; |
| struct list KeyTraceEventSink; |
| struct list PreservedKeyNotifySink; |
| struct list ThreadFocusSink; |
| struct list ThreadMgrEventSink; |
| } ThreadMgr; |
| |
| static inline ThreadMgr *impl_from_ITfSourceVtbl(ITfSource *iface) |
| { |
| return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,SourceVtbl)); |
| } |
| |
| static inline ThreadMgr *impl_from_ITfThreadMgrEventSink(ITfThreadMgrEventSink *iface) |
| { |
| return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,ThreadMgrEventSinkVtbl)); |
| } |
| |
| static void free_sink(ThreadMgrSink *sink) |
| { |
| IUnknown_Release(sink->interfaces.pIUnknown); |
| HeapFree(GetProcessHeap(),0,sink); |
| } |
| |
| static void ThreadMgr_Destructor(ThreadMgr *This) |
| { |
| struct list *cursor, *cursor2; |
| |
| TlsSetValue(tlsIndex,NULL); |
| TRACE("destroying %p\n", This); |
| if (This->focus) |
| ITfDocumentMgr_Release(This->focus); |
| |
| /* free sinks */ |
| LIST_FOR_EACH_SAFE(cursor, cursor2, &This->ActiveLanguageProfileNotifySink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| list_remove(cursor); |
| free_sink(sink); |
| } |
| LIST_FOR_EACH_SAFE(cursor, cursor2, &This->DisplayAttributeNotifySink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| list_remove(cursor); |
| free_sink(sink); |
| } |
| LIST_FOR_EACH_SAFE(cursor, cursor2, &This->KeyTraceEventSink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| list_remove(cursor); |
| free_sink(sink); |
| } |
| LIST_FOR_EACH_SAFE(cursor, cursor2, &This->PreservedKeyNotifySink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| list_remove(cursor); |
| free_sink(sink); |
| } |
| LIST_FOR_EACH_SAFE(cursor, cursor2, &This->ThreadFocusSink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| list_remove(cursor); |
| free_sink(sink); |
| } |
| LIST_FOR_EACH_SAFE(cursor, cursor2, &This->ThreadMgrEventSink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| list_remove(cursor); |
| free_sink(sink); |
| } |
| |
| HeapFree(GetProcessHeap(),0,This); |
| } |
| |
| static HRESULT WINAPI ThreadMgr_QueryInterface(ITfThreadMgr *iface, REFIID iid, LPVOID *ppvOut) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| *ppvOut = NULL; |
| |
| if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_ITfThreadMgr)) |
| { |
| *ppvOut = This; |
| } |
| else if (IsEqualIID(iid, &IID_ITfSource)) |
| { |
| *ppvOut = &This->SourceVtbl; |
| } |
| |
| if (*ppvOut) |
| { |
| IUnknown_AddRef(iface); |
| return S_OK; |
| } |
| |
| WARN("unsupported interface: %s\n", debugstr_guid(iid)); |
| return E_NOINTERFACE; |
| } |
| |
| static ULONG WINAPI ThreadMgr_AddRef(ITfThreadMgr *iface) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| return InterlockedIncrement(&This->refCount); |
| } |
| |
| static ULONG WINAPI ThreadMgr_Release(ITfThreadMgr *iface) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| ULONG ret; |
| |
| ret = InterlockedDecrement(&This->refCount); |
| if (ret == 0) |
| ThreadMgr_Destructor(This); |
| return ret; |
| } |
| |
| /***************************************************** |
| * ITfThreadMgr functions |
| *****************************************************/ |
| |
| static HRESULT WINAPI ThreadMgr_fnActivate( ITfThreadMgr* iface, TfClientId *ptid) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| FIXME("STUB:(%p)\n",This); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI ThreadMgr_fnDeactivate( ITfThreadMgr* iface) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| FIXME("STUB:(%p)\n",This); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI ThreadMgr_CreateDocumentMgr( ITfThreadMgr* iface, ITfDocumentMgr |
| **ppdim) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| TRACE("(%p)\n",iface); |
| return DocumentMgr_Constructor((ITfThreadMgrEventSink*)&This->ThreadMgrEventSinkVtbl, ppdim); |
| } |
| |
| static HRESULT WINAPI ThreadMgr_EnumDocumentMgrs( ITfThreadMgr* iface, IEnumTfDocumentMgrs |
| **ppEnum) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| FIXME("STUB:(%p)\n",This); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI ThreadMgr_GetFocus( ITfThreadMgr* iface, ITfDocumentMgr |
| **ppdimFocus) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| TRACE("(%p)\n",This); |
| |
| if (!ppdimFocus) |
| return E_INVALIDARG; |
| |
| *ppdimFocus = This->focus; |
| |
| TRACE("->%p\n",This->focus); |
| |
| if (This->focus == NULL) |
| return S_FALSE; |
| |
| ITfDocumentMgr_AddRef(This->focus); |
| |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI ThreadMgr_SetFocus( ITfThreadMgr* iface, ITfDocumentMgr *pdimFocus) |
| { |
| ITfDocumentMgr *check; |
| ThreadMgr *This = (ThreadMgr *)iface; |
| |
| TRACE("(%p) %p\n",This,pdimFocus); |
| |
| if (!pdimFocus || FAILED(IUnknown_QueryInterface(pdimFocus,&IID_ITfDocumentMgr,(LPVOID*) &check))) |
| return E_INVALIDARG; |
| |
| ITfThreadMgrEventSink_OnSetFocus((ITfThreadMgrEventSink*)&This->ThreadMgrEventSinkVtbl, This->focus, check); |
| |
| if (This->focus) |
| ITfDocumentMgr_Release(This->focus); |
| |
| This->focus = check; |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI ThreadMgr_AssociateFocus( ITfThreadMgr* iface, HWND hwnd, |
| ITfDocumentMgr *pdimNew, ITfDocumentMgr **ppdimPrev) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| FIXME("STUB:(%p)\n",This); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI ThreadMgr_IsThreadFocus( ITfThreadMgr* iface, BOOL *pfThreadFocus) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| FIXME("STUB:(%p)\n",This); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI ThreadMgr_GetFunctionProvider( ITfThreadMgr* iface, REFCLSID clsid, |
| ITfFunctionProvider **ppFuncProv) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| FIXME("STUB:(%p)\n",This); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI ThreadMgr_EnumFunctionProviders( ITfThreadMgr* iface, |
| IEnumTfFunctionProviders **ppEnum) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| FIXME("STUB:(%p)\n",This); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI ThreadMgr_GetGlobalCompartment( ITfThreadMgr* iface, |
| ITfCompartmentMgr **ppCompMgr) |
| { |
| ThreadMgr *This = (ThreadMgr *)iface; |
| FIXME("STUB:(%p)\n",This); |
| return E_NOTIMPL; |
| } |
| |
| static const ITfThreadMgrVtbl ThreadMgr_ThreadMgrVtbl = |
| { |
| ThreadMgr_QueryInterface, |
| ThreadMgr_AddRef, |
| ThreadMgr_Release, |
| |
| ThreadMgr_fnActivate, |
| ThreadMgr_fnDeactivate, |
| ThreadMgr_CreateDocumentMgr, |
| ThreadMgr_EnumDocumentMgrs, |
| ThreadMgr_GetFocus, |
| ThreadMgr_SetFocus, |
| ThreadMgr_AssociateFocus, |
| ThreadMgr_IsThreadFocus, |
| ThreadMgr_GetFunctionProvider, |
| ThreadMgr_EnumFunctionProviders, |
| ThreadMgr_GetGlobalCompartment |
| }; |
| |
| |
| static HRESULT WINAPI Source_QueryInterface(ITfSource *iface, REFIID iid, LPVOID *ppvOut) |
| { |
| ThreadMgr *This = impl_from_ITfSourceVtbl(iface); |
| return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut); |
| } |
| |
| static ULONG WINAPI Source_AddRef(ITfSource *iface) |
| { |
| ThreadMgr *This = impl_from_ITfSourceVtbl(iface); |
| return ThreadMgr_AddRef((ITfThreadMgr*)This); |
| } |
| |
| static ULONG WINAPI Source_Release(ITfSource *iface) |
| { |
| ThreadMgr *This = impl_from_ITfSourceVtbl(iface); |
| return ThreadMgr_Release((ITfThreadMgr *)This); |
| } |
| |
| /***************************************************** |
| * ITfSource functions |
| *****************************************************/ |
| static WINAPI HRESULT ThreadMgrSource_AdviseSink(ITfSource *iface, |
| REFIID riid, IUnknown *punk, DWORD *pdwCookie) |
| { |
| ThreadMgrSink *tms; |
| ThreadMgr *This = impl_from_ITfSourceVtbl(iface); |
| |
| TRACE("(%p) %s %p %p\n",This,debugstr_guid(riid),punk,pdwCookie); |
| |
| if (!riid || !punk || !pdwCookie) |
| return E_INVALIDARG; |
| |
| if (IsEqualIID(riid, &IID_ITfThreadMgrEventSink)) |
| { |
| tms = HeapAlloc(GetProcessHeap(),0,sizeof(ThreadMgrSink)); |
| if (!tms) |
| return E_OUTOFMEMORY; |
| if (!SUCCEEDED(IUnknown_QueryInterface(punk, riid, (LPVOID*)&tms->interfaces.pITfThreadMgrEventSink))) |
| { |
| HeapFree(GetProcessHeap(),0,tms); |
| return CONNECT_E_CANNOTCONNECT; |
| } |
| list_add_head(&This->ThreadMgrEventSink,&tms->entry); |
| *pdwCookie = (DWORD)tms; |
| } |
| else |
| { |
| FIXME("(%p) Unhandled Sink: %s\n",This,debugstr_guid(riid)); |
| return E_NOTIMPL; |
| } |
| |
| TRACE("cookie %x\n",*pdwCookie); |
| |
| return S_OK; |
| } |
| |
| static WINAPI HRESULT ThreadMgrSource_UnadviseSink(ITfSource *iface, DWORD pdwCookie) |
| { |
| ThreadMgrSink *sink = (ThreadMgrSink*)pdwCookie; |
| ThreadMgr *This = impl_from_ITfSourceVtbl(iface); |
| TRACE("(%p) %x\n",This,pdwCookie); |
| |
| list_remove(&sink->entry); |
| free_sink(sink); |
| |
| return S_OK; |
| } |
| |
| static const ITfSourceVtbl ThreadMgr_SourceVtbl = |
| { |
| Source_QueryInterface, |
| Source_AddRef, |
| Source_Release, |
| |
| ThreadMgrSource_AdviseSink, |
| ThreadMgrSource_UnadviseSink, |
| }; |
| |
| /***************************************************** |
| * ITfThreadMgrEventSink functions (internal) |
| *****************************************************/ |
| static HRESULT WINAPI ThreadMgrEventSink_QueryInterface(ITfThreadMgrEventSink *iface, REFIID iid, LPVOID *ppvOut) |
| { |
| ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface); |
| return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut); |
| } |
| |
| static ULONG WINAPI ThreadMgrEventSink_AddRef(ITfThreadMgrEventSink *iface) |
| { |
| ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface); |
| return ThreadMgr_AddRef((ITfThreadMgr*)This); |
| } |
| |
| static ULONG WINAPI ThreadMgrEventSink_Release(ITfThreadMgrEventSink *iface) |
| { |
| ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface); |
| return ThreadMgr_Release((ITfThreadMgr *)This); |
| } |
| |
| |
| static WINAPI HRESULT ThreadMgrEventSink_OnInitDocumentMgr( |
| ITfThreadMgrEventSink *iface,ITfDocumentMgr *pdim) |
| { |
| struct list *cursor; |
| ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface); |
| |
| TRACE("(%p) %p\n",This,pdim); |
| |
| LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| ITfThreadMgrEventSink_OnInitDocumentMgr(sink->interfaces.pITfThreadMgrEventSink,pdim); |
| } |
| |
| return S_OK; |
| } |
| |
| static WINAPI HRESULT ThreadMgrEventSink_OnUninitDocumentMgr( |
| ITfThreadMgrEventSink *iface, ITfDocumentMgr *pdim) |
| { |
| struct list *cursor; |
| ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface); |
| |
| TRACE("(%p) %p\n",This,pdim); |
| |
| LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| ITfThreadMgrEventSink_OnUninitDocumentMgr(sink->interfaces.pITfThreadMgrEventSink,pdim); |
| } |
| |
| return S_OK; |
| } |
| |
| static WINAPI HRESULT ThreadMgrEventSink_OnSetFocus( |
| ITfThreadMgrEventSink *iface, ITfDocumentMgr *pdimFocus, |
| ITfDocumentMgr *pdimPrevFocus) |
| { |
| struct list *cursor; |
| ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface); |
| |
| TRACE("(%p) %p %p\n",This,pdimFocus, pdimPrevFocus); |
| |
| LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| ITfThreadMgrEventSink_OnSetFocus(sink->interfaces.pITfThreadMgrEventSink, pdimFocus, pdimPrevFocus); |
| } |
| |
| return S_OK; |
| } |
| |
| static WINAPI HRESULT ThreadMgrEventSink_OnPushContext( |
| ITfThreadMgrEventSink *iface, ITfContext *pic) |
| { |
| struct list *cursor; |
| ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface); |
| |
| TRACE("(%p) %p\n",This,pic); |
| |
| LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| ITfThreadMgrEventSink_OnPushContext(sink->interfaces.pITfThreadMgrEventSink,pic); |
| } |
| |
| return S_OK; |
| } |
| |
| static WINAPI HRESULT ThreadMgrEventSink_OnPopContext( |
| ITfThreadMgrEventSink *iface, ITfContext *pic) |
| { |
| struct list *cursor; |
| ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface); |
| |
| TRACE("(%p) %p\n",This,pic); |
| |
| LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink) |
| { |
| ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry); |
| ITfThreadMgrEventSink_OnPopContext(sink->interfaces.pITfThreadMgrEventSink,pic); |
| } |
| |
| return S_OK; |
| } |
| |
| static const ITfThreadMgrEventSinkVtbl ThreadMgr_ThreadMgrEventSinkVtbl = |
| { |
| ThreadMgrEventSink_QueryInterface, |
| ThreadMgrEventSink_AddRef, |
| ThreadMgrEventSink_Release, |
| |
| ThreadMgrEventSink_OnInitDocumentMgr, |
| ThreadMgrEventSink_OnUninitDocumentMgr, |
| ThreadMgrEventSink_OnSetFocus, |
| ThreadMgrEventSink_OnPushContext, |
| ThreadMgrEventSink_OnPopContext |
| }; |
| |
| HRESULT ThreadMgr_Constructor(IUnknown *pUnkOuter, IUnknown **ppOut) |
| { |
| ThreadMgr *This; |
| if (pUnkOuter) |
| return CLASS_E_NOAGGREGATION; |
| |
| /* Only 1 ThreadMgr is created per thread */ |
| This = TlsGetValue(tlsIndex); |
| if (This) |
| { |
| ThreadMgr_AddRef((ITfThreadMgr*)This); |
| *ppOut = (IUnknown*)This; |
| return S_OK; |
| } |
| |
| This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(ThreadMgr)); |
| if (This == NULL) |
| return E_OUTOFMEMORY; |
| |
| This->ThreadMgrVtbl= &ThreadMgr_ThreadMgrVtbl; |
| This->SourceVtbl = &ThreadMgr_SourceVtbl; |
| This->ThreadMgrEventSinkVtbl = &ThreadMgr_ThreadMgrEventSinkVtbl; |
| This->refCount = 1; |
| TlsSetValue(tlsIndex,This); |
| |
| list_init(&This->ActiveLanguageProfileNotifySink); |
| list_init(&This->DisplayAttributeNotifySink); |
| list_init(&This->KeyTraceEventSink); |
| list_init(&This->PreservedKeyNotifySink); |
| list_init(&This->ThreadFocusSink); |
| list_init(&This->ThreadMgrEventSink); |
| |
| TRACE("returning %p\n", This); |
| *ppOut = (IUnknown *)This; |
| return S_OK; |
| } |