2 * ITfThreadMgr implementation
4 * Copyright 2008 Aric Stewart, CodeWeavers
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
27 #include "wine/debug.h"
37 #include "wine/unicode.h"
38 #include "wine/list.h"
41 #include "msctf_internal.h"
43 WINE_DEFAULT_DEBUG_CHANNEL(msctf);
45 typedef struct tagThreadMgrSink {
50 /* ITfActiveLanguageProfileNotifySink *pITfActiveLanguageProfileNotifySink; */
51 /* ITfDisplayAttributeNotifySink *pITfDisplayAttributeNotifySink; */
52 /* ITfKeyTraceEventSink *pITfKeyTraceEventSink; */
53 /* ITfPreservedKeyNotifySink *pITfPreservedKeyNotifySink; */
54 /* ITfThreadFocusSink *pITfThreadFocusSink; */
55 ITfThreadMgrEventSink *pITfThreadMgrEventSink;
59 typedef struct tagACLMulti {
60 const ITfThreadMgrVtbl *ThreadMgrVtbl;
61 const ITfSourceVtbl *SourceVtbl;
64 const ITfThreadMgrEventSinkVtbl *ThreadMgrEventSinkVtbl; /* internal */
66 ITfDocumentMgr *focus;
68 /* kept as separate lists to reduce unnecessary iterations */
69 struct list ActiveLanguageProfileNotifySink;
70 struct list DisplayAttributeNotifySink;
71 struct list KeyTraceEventSink;
72 struct list PreservedKeyNotifySink;
73 struct list ThreadFocusSink;
74 struct list ThreadMgrEventSink;
77 static inline ThreadMgr *impl_from_ITfSourceVtbl(ITfSource *iface)
79 return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,SourceVtbl));
82 static inline ThreadMgr *impl_from_ITfThreadMgrEventSink(ITfThreadMgrEventSink *iface)
84 return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,ThreadMgrEventSinkVtbl));
87 static void free_sink(ThreadMgrSink *sink)
89 IUnknown_Release(sink->interfaces.pIUnknown);
90 HeapFree(GetProcessHeap(),0,sink);
93 static void ThreadMgr_Destructor(ThreadMgr *This)
95 struct list *cursor, *cursor2;
97 TlsSetValue(tlsIndex,NULL);
98 TRACE("destroying %p\n", This);
100 ITfDocumentMgr_Release(This->focus);
103 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->ActiveLanguageProfileNotifySink)
105 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
109 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->DisplayAttributeNotifySink)
111 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
115 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->KeyTraceEventSink)
117 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
121 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->PreservedKeyNotifySink)
123 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
127 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->ThreadFocusSink)
129 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
133 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->ThreadMgrEventSink)
135 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
140 HeapFree(GetProcessHeap(),0,This);
143 static HRESULT WINAPI ThreadMgr_QueryInterface(ITfThreadMgr *iface, REFIID iid, LPVOID *ppvOut)
145 ThreadMgr *This = (ThreadMgr *)iface;
148 if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_ITfThreadMgr))
152 else if (IsEqualIID(iid, &IID_ITfSource))
154 *ppvOut = &This->SourceVtbl;
159 IUnknown_AddRef(iface);
163 WARN("unsupported interface: %s\n", debugstr_guid(iid));
164 return E_NOINTERFACE;
167 static ULONG WINAPI ThreadMgr_AddRef(ITfThreadMgr *iface)
169 ThreadMgr *This = (ThreadMgr *)iface;
170 return InterlockedIncrement(&This->refCount);
173 static ULONG WINAPI ThreadMgr_Release(ITfThreadMgr *iface)
175 ThreadMgr *This = (ThreadMgr *)iface;
178 ret = InterlockedDecrement(&This->refCount);
180 ThreadMgr_Destructor(This);
184 /*****************************************************
185 * ITfThreadMgr functions
186 *****************************************************/
188 static HRESULT WINAPI ThreadMgr_fnActivate( ITfThreadMgr* iface, TfClientId *ptid)
190 ThreadMgr *This = (ThreadMgr *)iface;
191 FIXME("STUB:(%p)\n",This);
195 static HRESULT WINAPI ThreadMgr_fnDeactivate( ITfThreadMgr* iface)
197 ThreadMgr *This = (ThreadMgr *)iface;
198 FIXME("STUB:(%p)\n",This);
202 static HRESULT WINAPI ThreadMgr_CreateDocumentMgr( ITfThreadMgr* iface, ITfDocumentMgr
205 ThreadMgr *This = (ThreadMgr *)iface;
206 TRACE("(%p)\n",iface);
207 return DocumentMgr_Constructor((ITfThreadMgrEventSink*)&This->ThreadMgrEventSinkVtbl, ppdim);
210 static HRESULT WINAPI ThreadMgr_EnumDocumentMgrs( ITfThreadMgr* iface, IEnumTfDocumentMgrs
213 ThreadMgr *This = (ThreadMgr *)iface;
214 FIXME("STUB:(%p)\n",This);
218 static HRESULT WINAPI ThreadMgr_GetFocus( ITfThreadMgr* iface, ITfDocumentMgr
221 ThreadMgr *This = (ThreadMgr *)iface;
222 TRACE("(%p)\n",This);
227 *ppdimFocus = This->focus;
229 TRACE("->%p\n",This->focus);
231 if (This->focus == NULL)
234 ITfDocumentMgr_AddRef(This->focus);
239 static HRESULT WINAPI ThreadMgr_SetFocus( ITfThreadMgr* iface, ITfDocumentMgr *pdimFocus)
241 ITfDocumentMgr *check;
242 ThreadMgr *This = (ThreadMgr *)iface;
244 TRACE("(%p) %p\n",This,pdimFocus);
246 if (!pdimFocus || FAILED(IUnknown_QueryInterface(pdimFocus,&IID_ITfDocumentMgr,(LPVOID*) &check)))
249 ITfThreadMgrEventSink_OnSetFocus((ITfThreadMgrEventSink*)&This->ThreadMgrEventSinkVtbl, This->focus, check);
252 ITfDocumentMgr_Release(This->focus);
258 static HRESULT WINAPI ThreadMgr_AssociateFocus( ITfThreadMgr* iface, HWND hwnd,
259 ITfDocumentMgr *pdimNew, ITfDocumentMgr **ppdimPrev)
261 ThreadMgr *This = (ThreadMgr *)iface;
262 FIXME("STUB:(%p)\n",This);
266 static HRESULT WINAPI ThreadMgr_IsThreadFocus( ITfThreadMgr* iface, BOOL *pfThreadFocus)
268 ThreadMgr *This = (ThreadMgr *)iface;
269 FIXME("STUB:(%p)\n",This);
273 static HRESULT WINAPI ThreadMgr_GetFunctionProvider( ITfThreadMgr* iface, REFCLSID clsid,
274 ITfFunctionProvider **ppFuncProv)
276 ThreadMgr *This = (ThreadMgr *)iface;
277 FIXME("STUB:(%p)\n",This);
281 static HRESULT WINAPI ThreadMgr_EnumFunctionProviders( ITfThreadMgr* iface,
282 IEnumTfFunctionProviders **ppEnum)
284 ThreadMgr *This = (ThreadMgr *)iface;
285 FIXME("STUB:(%p)\n",This);
289 static HRESULT WINAPI ThreadMgr_GetGlobalCompartment( ITfThreadMgr* iface,
290 ITfCompartmentMgr **ppCompMgr)
292 ThreadMgr *This = (ThreadMgr *)iface;
293 FIXME("STUB:(%p)\n",This);
297 static const ITfThreadMgrVtbl ThreadMgr_ThreadMgrVtbl =
299 ThreadMgr_QueryInterface,
303 ThreadMgr_fnActivate,
304 ThreadMgr_fnDeactivate,
305 ThreadMgr_CreateDocumentMgr,
306 ThreadMgr_EnumDocumentMgrs,
309 ThreadMgr_AssociateFocus,
310 ThreadMgr_IsThreadFocus,
311 ThreadMgr_GetFunctionProvider,
312 ThreadMgr_EnumFunctionProviders,
313 ThreadMgr_GetGlobalCompartment
317 static HRESULT WINAPI Source_QueryInterface(ITfSource *iface, REFIID iid, LPVOID *ppvOut)
319 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
320 return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut);
323 static ULONG WINAPI Source_AddRef(ITfSource *iface)
325 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
326 return ThreadMgr_AddRef((ITfThreadMgr*)This);
329 static ULONG WINAPI Source_Release(ITfSource *iface)
331 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
332 return ThreadMgr_Release((ITfThreadMgr *)This);
335 /*****************************************************
336 * ITfSource functions
337 *****************************************************/
338 static WINAPI HRESULT ThreadMgrSource_AdviseSink(ITfSource *iface,
339 REFIID riid, IUnknown *punk, DWORD *pdwCookie)
342 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
344 TRACE("(%p) %s %p %p\n",This,debugstr_guid(riid),punk,pdwCookie);
346 if (!riid || !punk || !pdwCookie)
349 if (IsEqualIID(riid, &IID_ITfThreadMgrEventSink))
351 tms = HeapAlloc(GetProcessHeap(),0,sizeof(ThreadMgrSink));
353 return E_OUTOFMEMORY;
354 if (!SUCCEEDED(IUnknown_QueryInterface(punk, riid, (LPVOID*)&tms->interfaces.pITfThreadMgrEventSink)))
356 HeapFree(GetProcessHeap(),0,tms);
357 return CONNECT_E_CANNOTCONNECT;
359 list_add_head(&This->ThreadMgrEventSink,&tms->entry);
360 *pdwCookie = (DWORD)tms;
364 FIXME("(%p) Unhandled Sink: %s\n",This,debugstr_guid(riid));
368 TRACE("cookie %x\n",*pdwCookie);
373 static WINAPI HRESULT ThreadMgrSource_UnadviseSink(ITfSource *iface, DWORD pdwCookie)
375 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
376 FIXME("STUB:(%p)\n",This);
380 static const ITfSourceVtbl ThreadMgr_SourceVtbl =
382 Source_QueryInterface,
386 ThreadMgrSource_AdviseSink,
387 ThreadMgrSource_UnadviseSink,
390 /*****************************************************
391 * ITfThreadMgrEventSink functions (internal)
392 *****************************************************/
393 static HRESULT WINAPI ThreadMgrEventSink_QueryInterface(ITfThreadMgrEventSink *iface, REFIID iid, LPVOID *ppvOut)
395 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
396 return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut);
399 static ULONG WINAPI ThreadMgrEventSink_AddRef(ITfThreadMgrEventSink *iface)
401 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
402 return ThreadMgr_AddRef((ITfThreadMgr*)This);
405 static ULONG WINAPI ThreadMgrEventSink_Release(ITfThreadMgrEventSink *iface)
407 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
408 return ThreadMgr_Release((ITfThreadMgr *)This);
412 static WINAPI HRESULT ThreadMgrEventSink_OnInitDocumentMgr(
413 ITfThreadMgrEventSink *iface,ITfDocumentMgr *pdim)
416 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
418 TRACE("(%p) %p\n",This,pdim);
420 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
422 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
423 ITfThreadMgrEventSink_OnInitDocumentMgr(sink->interfaces.pITfThreadMgrEventSink,pdim);
429 static WINAPI HRESULT ThreadMgrEventSink_OnUninitDocumentMgr(
430 ITfThreadMgrEventSink *iface, ITfDocumentMgr *pdim)
433 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
435 TRACE("(%p) %p\n",This,pdim);
437 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
439 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
440 ITfThreadMgrEventSink_OnUninitDocumentMgr(sink->interfaces.pITfThreadMgrEventSink,pdim);
446 static WINAPI HRESULT ThreadMgrEventSink_OnSetFocus(
447 ITfThreadMgrEventSink *iface, ITfDocumentMgr *pdimFocus,
448 ITfDocumentMgr *pdimPrevFocus)
451 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
453 TRACE("(%p) %p %p\n",This,pdimFocus, pdimPrevFocus);
455 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
457 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
458 ITfThreadMgrEventSink_OnSetFocus(sink->interfaces.pITfThreadMgrEventSink, pdimFocus, pdimPrevFocus);
464 static WINAPI HRESULT ThreadMgrEventSink_OnPushContext(
465 ITfThreadMgrEventSink *iface, ITfContext *pic)
468 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
470 TRACE("(%p) %p\n",This,pic);
472 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
474 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
475 ITfThreadMgrEventSink_OnPushContext(sink->interfaces.pITfThreadMgrEventSink,pic);
481 static WINAPI HRESULT ThreadMgrEventSink_OnPopContext(
482 ITfThreadMgrEventSink *iface, ITfContext *pic)
485 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
487 TRACE("(%p) %p\n",This,pic);
489 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
491 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
492 ITfThreadMgrEventSink_OnPopContext(sink->interfaces.pITfThreadMgrEventSink,pic);
498 static const ITfThreadMgrEventSinkVtbl ThreadMgr_ThreadMgrEventSinkVtbl =
500 ThreadMgrEventSink_QueryInterface,
501 ThreadMgrEventSink_AddRef,
502 ThreadMgrEventSink_Release,
504 ThreadMgrEventSink_OnInitDocumentMgr,
505 ThreadMgrEventSink_OnUninitDocumentMgr,
506 ThreadMgrEventSink_OnSetFocus,
507 ThreadMgrEventSink_OnPushContext,
508 ThreadMgrEventSink_OnPopContext
511 HRESULT ThreadMgr_Constructor(IUnknown *pUnkOuter, IUnknown **ppOut)
515 return CLASS_E_NOAGGREGATION;
517 /* Only 1 ThreadMgr is created per thread */
518 This = TlsGetValue(tlsIndex);
521 ThreadMgr_AddRef((ITfThreadMgr*)This);
522 *ppOut = (IUnknown*)This;
526 This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(ThreadMgr));
528 return E_OUTOFMEMORY;
530 This->ThreadMgrVtbl= &ThreadMgr_ThreadMgrVtbl;
531 This->SourceVtbl = &ThreadMgr_SourceVtbl;
532 This->ThreadMgrEventSinkVtbl = &ThreadMgr_ThreadMgrEventSinkVtbl;
534 TlsSetValue(tlsIndex,This);
536 list_init(&This->ActiveLanguageProfileNotifySink);
537 list_init(&This->DisplayAttributeNotifySink);
538 list_init(&This->KeyTraceEventSink);
539 list_init(&This->PreservedKeyNotifySink);
540 list_init(&This->ThreadFocusSink);
541 list_init(&This->ThreadMgrEventSink);
543 TRACE("returning %p\n", This);
544 *ppOut = (IUnknown *)This;