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 tagPreservedKey
63 TF_PRESERVEDKEY prekey;
68 typedef struct tagDocumentMgrs
71 ITfDocumentMgr *docmgr;
74 typedef struct tagAssociatedWindow
78 ITfDocumentMgr *docmgr;
81 typedef struct tagACLMulti {
82 const ITfThreadMgrVtbl *ThreadMgrVtbl;
83 const ITfSourceVtbl *SourceVtbl;
84 const ITfKeystrokeMgrVtbl *KeystrokeMgrVtbl;
85 const ITfMessagePumpVtbl *MessagePumpVtbl;
86 const ITfClientIdVtbl *ClientIdVtbl;
87 /* const ITfThreadMgrExVtbl *ThreadMgrExVtbl; */
88 /* const ITfConfigureSystemKeystrokeFeedVtbl *ConfigureSystemKeystrokeFeedVtbl; */
89 /* const ITfLangBarItemMgrVtbl *LangBarItemMgrVtbl; */
90 /* const ITfUIElementMgrVtbl *UIElementMgrVtbl; */
91 const ITfSourceSingleVtbl *SourceSingleVtbl;
95 ITfCompartmentMgr *CompartmentMgr;
97 const ITfThreadMgrEventSinkVtbl *ThreadMgrEventSinkVtbl; /* internal */
99 ITfDocumentMgr *focus;
100 LONG activationCount;
102 ITfKeyEventSink *forgroundKeyEventSink;
103 CLSID forgroundTextService;
105 struct list CurrentPreservedKeys;
106 struct list CreatedDocumentMgrs;
108 struct list AssociatedFocusWindows;
111 /* kept as separate lists to reduce unnecessary iterations */
112 struct list ActiveLanguageProfileNotifySink;
113 struct list DisplayAttributeNotifySink;
114 struct list KeyTraceEventSink;
115 struct list PreservedKeyNotifySink;
116 struct list ThreadFocusSink;
117 struct list ThreadMgrEventSink;
120 typedef struct tagEnumTfDocumentMgr {
121 const IEnumTfDocumentMgrsVtbl *Vtbl;
128 static HRESULT EnumTfDocumentMgr_Constructor(struct list* head, IEnumTfDocumentMgrs **ppOut);
129 LRESULT CALLBACK ThreadFocusHookProc(int nCode, WPARAM wParam, LPARAM lParam);
131 static inline ThreadMgr *impl_from_ITfSourceVtbl(ITfSource *iface)
133 return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,SourceVtbl));
136 static inline ThreadMgr *impl_from_ITfKeystrokeMgrVtbl(ITfKeystrokeMgr *iface)
138 return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,KeystrokeMgrVtbl));
141 static inline ThreadMgr *impl_from_ITfMessagePumpVtbl(ITfMessagePump *iface)
143 return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,MessagePumpVtbl));
146 static inline ThreadMgr *impl_from_ITfClientIdVtbl(ITfClientId *iface)
148 return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,ClientIdVtbl));
151 static inline ThreadMgr *impl_from_ITfThreadMgrEventSink(ITfThreadMgrEventSink *iface)
153 return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,ThreadMgrEventSinkVtbl));
156 static inline ThreadMgr *impl_from_ITfSourceSingleVtbl(ITfSourceSingle* iface)
159 return (ThreadMgr *)((char *)iface - FIELD_OFFSET(ThreadMgr,SourceSingleVtbl));
162 static HRESULT SetupWindowsHook(ThreadMgr *This)
164 if (!This->focusHook)
166 This->focusHook = SetWindowsHookExW(WH_CBT, ThreadFocusHookProc, 0,
167 GetCurrentThreadId());
168 if (!This->focusHook)
170 ERR("Unable to set focus hook\n");
178 static void free_sink(ThreadMgrSink *sink)
180 IUnknown_Release(sink->interfaces.pIUnknown);
181 HeapFree(GetProcessHeap(),0,sink);
184 static void ThreadMgr_Destructor(ThreadMgr *This)
186 struct list *cursor, *cursor2;
188 /* unhook right away */
190 UnhookWindowsHookEx(This->focusHook);
192 TlsSetValue(tlsIndex,NULL);
193 TRACE("destroying %p\n", This);
195 ITfDocumentMgr_Release(This->focus);
198 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->ActiveLanguageProfileNotifySink)
200 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
204 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->DisplayAttributeNotifySink)
206 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
210 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->KeyTraceEventSink)
212 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
216 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->PreservedKeyNotifySink)
218 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
222 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->ThreadFocusSink)
224 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
228 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->ThreadMgrEventSink)
230 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
235 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->CurrentPreservedKeys)
237 PreservedKey* key = LIST_ENTRY(cursor,PreservedKey,entry);
239 HeapFree(GetProcessHeap(),0,key->description);
240 HeapFree(GetProcessHeap(),0,key);
243 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->CreatedDocumentMgrs)
245 DocumentMgrEntry *mgr = LIST_ENTRY(cursor,DocumentMgrEntry,entry);
247 FIXME("Left Over ITfDocumentMgr. Should we do something with it?\n");
248 HeapFree(GetProcessHeap(),0,mgr);
251 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->AssociatedFocusWindows)
253 AssociatedWindow *wnd = LIST_ENTRY(cursor,AssociatedWindow,entry);
255 HeapFree(GetProcessHeap(),0,wnd);
258 CompartmentMgr_Destructor(This->CompartmentMgr);
260 HeapFree(GetProcessHeap(),0,This);
263 static HRESULT WINAPI ThreadMgr_QueryInterface(ITfThreadMgr *iface, REFIID iid, LPVOID *ppvOut)
265 ThreadMgr *This = (ThreadMgr *)iface;
268 if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_ITfThreadMgr))
272 else if (IsEqualIID(iid, &IID_ITfSource))
274 *ppvOut = &This->SourceVtbl;
276 else if (IsEqualIID(iid, &IID_ITfKeystrokeMgr))
278 *ppvOut = &This->KeystrokeMgrVtbl;
280 else if (IsEqualIID(iid, &IID_ITfMessagePump))
282 *ppvOut = &This->MessagePumpVtbl;
284 else if (IsEqualIID(iid, &IID_ITfClientId))
286 *ppvOut = &This->ClientIdVtbl;
288 else if (IsEqualIID(iid, &IID_ITfCompartmentMgr))
290 *ppvOut = This->CompartmentMgr;
292 else if (IsEqualIID(iid, &IID_ITfSourceSingle))
294 *ppvOut = &This->SourceSingleVtbl;
299 IUnknown_AddRef(iface);
303 WARN("unsupported interface: %s\n", debugstr_guid(iid));
304 return E_NOINTERFACE;
307 static ULONG WINAPI ThreadMgr_AddRef(ITfThreadMgr *iface)
309 ThreadMgr *This = (ThreadMgr *)iface;
310 return InterlockedIncrement(&This->refCount);
313 static ULONG WINAPI ThreadMgr_Release(ITfThreadMgr *iface)
315 ThreadMgr *This = (ThreadMgr *)iface;
318 ret = InterlockedDecrement(&This->refCount);
320 ThreadMgr_Destructor(This);
324 /*****************************************************
325 * ITfThreadMgr functions
326 *****************************************************/
328 static HRESULT WINAPI ThreadMgr_fnActivate( ITfThreadMgr* iface, TfClientId *ptid)
330 ThreadMgr *This = (ThreadMgr *)iface;
332 TRACE("(%p) %p\n",This, ptid);
341 ITfClientId_GetClientId((ITfClientId*)&This->ClientIdVtbl,&guid,&processId);
344 activate_textservices(iface);
345 This->activationCount++;
350 static HRESULT WINAPI ThreadMgr_fnDeactivate( ITfThreadMgr* iface)
352 ThreadMgr *This = (ThreadMgr *)iface;
353 TRACE("(%p)\n",This);
355 if (This->activationCount == 0)
358 This->activationCount --;
360 if (This->activationCount == 0)
364 ITfThreadMgrEventSink_OnSetFocus((ITfThreadMgrEventSink*)&This->ThreadMgrEventSinkVtbl, 0, This->focus);
365 ITfDocumentMgr_Release(This->focus);
370 deactivate_textservices();
375 static HRESULT WINAPI ThreadMgr_CreateDocumentMgr( ITfThreadMgr* iface, ITfDocumentMgr
378 ThreadMgr *This = (ThreadMgr *)iface;
379 DocumentMgrEntry *mgrentry;
382 TRACE("(%p)\n",iface);
383 mgrentry = HeapAlloc(GetProcessHeap(),0,sizeof(DocumentMgrEntry));
384 if (mgrentry == NULL)
385 return E_OUTOFMEMORY;
387 hr = DocumentMgr_Constructor((ITfThreadMgrEventSink*)&This->ThreadMgrEventSinkVtbl, ppdim);
391 mgrentry->docmgr = *ppdim;
392 list_add_head(&This->CreatedDocumentMgrs,&mgrentry->entry);
395 HeapFree(GetProcessHeap(),0,mgrentry);
400 static HRESULT WINAPI ThreadMgr_EnumDocumentMgrs( ITfThreadMgr* iface, IEnumTfDocumentMgrs
403 ThreadMgr *This = (ThreadMgr *)iface;
404 TRACE("(%p) %p\n",This,ppEnum);
409 return EnumTfDocumentMgr_Constructor(&This->CreatedDocumentMgrs, ppEnum);
412 static HRESULT WINAPI ThreadMgr_GetFocus( ITfThreadMgr* iface, ITfDocumentMgr
415 ThreadMgr *This = (ThreadMgr *)iface;
416 TRACE("(%p)\n",This);
421 *ppdimFocus = This->focus;
423 TRACE("->%p\n",This->focus);
425 if (This->focus == NULL)
428 ITfDocumentMgr_AddRef(This->focus);
433 static HRESULT WINAPI ThreadMgr_SetFocus( ITfThreadMgr* iface, ITfDocumentMgr *pdimFocus)
435 ITfDocumentMgr *check;
436 ThreadMgr *This = (ThreadMgr *)iface;
438 TRACE("(%p) %p\n",This,pdimFocus);
442 else if (FAILED(IUnknown_QueryInterface(pdimFocus,&IID_ITfDocumentMgr,(LPVOID*) &check)))
445 ITfThreadMgrEventSink_OnSetFocus((ITfThreadMgrEventSink*)&This->ThreadMgrEventSinkVtbl, check, This->focus);
448 ITfDocumentMgr_Release(This->focus);
454 static HRESULT WINAPI ThreadMgr_AssociateFocus( ITfThreadMgr* iface, HWND hwnd,
455 ITfDocumentMgr *pdimNew, ITfDocumentMgr **ppdimPrev)
457 struct list *cursor, *cursor2;
458 ThreadMgr *This = (ThreadMgr *)iface;
459 AssociatedWindow *wnd;
461 TRACE("(%p) %p %p %p\n",This,hwnd,pdimNew,ppdimPrev);
468 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->AssociatedFocusWindows)
470 wnd = LIST_ENTRY(cursor,AssociatedWindow,entry);
471 if (wnd->hwnd == hwnd)
474 ITfDocumentMgr_AddRef(wnd->docmgr);
475 *ppdimPrev = wnd->docmgr;
476 wnd->docmgr = pdimNew;
477 if (GetFocus() == hwnd)
478 ThreadMgr_SetFocus(iface,pdimNew);
483 wnd = HeapAlloc(GetProcessHeap(),0,sizeof(AssociatedWindow));
485 wnd->docmgr = pdimNew;
486 list_add_head(&This->AssociatedFocusWindows,&wnd->entry);
488 if (GetFocus() == hwnd)
489 ThreadMgr_SetFocus(iface,pdimNew);
491 SetupWindowsHook(This);
496 static HRESULT WINAPI ThreadMgr_IsThreadFocus( ITfThreadMgr* iface, BOOL *pfThreadFocus)
499 ThreadMgr *This = (ThreadMgr *)iface;
500 TRACE("(%p) %p\n",This,pfThreadFocus);
502 *pfThreadFocus = (focus == NULL);
506 static HRESULT WINAPI ThreadMgr_GetFunctionProvider( ITfThreadMgr* iface, REFCLSID clsid,
507 ITfFunctionProvider **ppFuncProv)
509 ThreadMgr *This = (ThreadMgr *)iface;
510 FIXME("STUB:(%p)\n",This);
514 static HRESULT WINAPI ThreadMgr_EnumFunctionProviders( ITfThreadMgr* iface,
515 IEnumTfFunctionProviders **ppEnum)
517 ThreadMgr *This = (ThreadMgr *)iface;
518 FIXME("STUB:(%p)\n",This);
522 static HRESULT WINAPI ThreadMgr_GetGlobalCompartment( ITfThreadMgr* iface,
523 ITfCompartmentMgr **ppCompMgr)
525 ThreadMgr *This = (ThreadMgr *)iface;
527 TRACE("(%p) %p\n",This, ppCompMgr);
532 if (!globalCompartmentMgr)
534 hr = CompartmentMgr_Constructor(NULL,&IID_ITfCompartmentMgr,(IUnknown**)&globalCompartmentMgr);
539 ITfCompartmentMgr_AddRef(globalCompartmentMgr);
540 *ppCompMgr = globalCompartmentMgr;
544 static const ITfThreadMgrVtbl ThreadMgr_ThreadMgrVtbl =
546 ThreadMgr_QueryInterface,
550 ThreadMgr_fnActivate,
551 ThreadMgr_fnDeactivate,
552 ThreadMgr_CreateDocumentMgr,
553 ThreadMgr_EnumDocumentMgrs,
556 ThreadMgr_AssociateFocus,
557 ThreadMgr_IsThreadFocus,
558 ThreadMgr_GetFunctionProvider,
559 ThreadMgr_EnumFunctionProviders,
560 ThreadMgr_GetGlobalCompartment
564 static HRESULT WINAPI Source_QueryInterface(ITfSource *iface, REFIID iid, LPVOID *ppvOut)
566 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
567 return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut);
570 static ULONG WINAPI Source_AddRef(ITfSource *iface)
572 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
573 return ThreadMgr_AddRef((ITfThreadMgr*)This);
576 static ULONG WINAPI Source_Release(ITfSource *iface)
578 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
579 return ThreadMgr_Release((ITfThreadMgr *)This);
582 /*****************************************************
583 * ITfSource functions
584 *****************************************************/
585 static WINAPI HRESULT ThreadMgrSource_AdviseSink(ITfSource *iface,
586 REFIID riid, IUnknown *punk, DWORD *pdwCookie)
589 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
591 TRACE("(%p) %s %p %p\n",This,debugstr_guid(riid),punk,pdwCookie);
593 if (!riid || !punk || !pdwCookie)
596 if (IsEqualIID(riid, &IID_ITfThreadMgrEventSink))
598 tms = HeapAlloc(GetProcessHeap(),0,sizeof(ThreadMgrSink));
600 return E_OUTOFMEMORY;
601 if (FAILED(IUnknown_QueryInterface(punk, riid, (LPVOID *)&tms->interfaces.pITfThreadMgrEventSink)))
603 HeapFree(GetProcessHeap(),0,tms);
604 return CONNECT_E_CANNOTCONNECT;
606 list_add_head(&This->ThreadMgrEventSink,&tms->entry);
607 *pdwCookie = generate_Cookie(COOKIE_MAGIC_TMSINK, tms);
611 FIXME("(%p) Unhandled Sink: %s\n",This,debugstr_guid(riid));
615 TRACE("cookie %x\n",*pdwCookie);
620 static WINAPI HRESULT ThreadMgrSource_UnadviseSink(ITfSource *iface, DWORD pdwCookie)
623 ThreadMgr *This = impl_from_ITfSourceVtbl(iface);
625 TRACE("(%p) %x\n",This,pdwCookie);
627 if (get_Cookie_magic(pdwCookie)!=COOKIE_MAGIC_TMSINK)
630 sink = (ThreadMgrSink*)remove_Cookie(pdwCookie);
632 return CONNECT_E_NOCONNECTION;
634 list_remove(&sink->entry);
640 static const ITfSourceVtbl ThreadMgr_SourceVtbl =
642 Source_QueryInterface,
646 ThreadMgrSource_AdviseSink,
647 ThreadMgrSource_UnadviseSink,
650 /*****************************************************
651 * ITfKeystrokeMgr functions
652 *****************************************************/
654 static HRESULT WINAPI KeystrokeMgr_QueryInterface(ITfKeystrokeMgr *iface, REFIID iid, LPVOID *ppvOut)
656 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
657 return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut);
660 static ULONG WINAPI KeystrokeMgr_AddRef(ITfKeystrokeMgr *iface)
662 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
663 return ThreadMgr_AddRef((ITfThreadMgr*)This);
666 static ULONG WINAPI KeystrokeMgr_Release(ITfKeystrokeMgr *iface)
668 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
669 return ThreadMgr_Release((ITfThreadMgr *)This);
672 static HRESULT WINAPI KeystrokeMgr_AdviseKeyEventSink(ITfKeystrokeMgr *iface,
673 TfClientId tid, ITfKeyEventSink *pSink, BOOL fForeground)
675 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
677 ITfKeyEventSink *check = NULL;
679 TRACE("(%p) %x %p %i\n",This,tid,pSink,fForeground);
684 textservice = get_textservice_clsid(tid);
685 if (IsEqualCLSID(&GUID_NULL,&textservice))
688 get_textservice_sink(tid, &IID_ITfKeyEventSink, (IUnknown**)&check);
690 return CONNECT_E_ADVISELIMIT;
692 if (FAILED(IUnknown_QueryInterface(pSink,&IID_ITfKeyEventSink,(LPVOID*) &check)))
695 set_textservice_sink(tid, &IID_ITfKeyEventSink, (IUnknown*)check);
699 if (This->forgroundKeyEventSink)
701 ITfKeyEventSink_OnSetFocus(This->forgroundKeyEventSink, FALSE);
702 ITfKeyEventSink_Release(This->forgroundKeyEventSink);
704 ITfKeyEventSink_AddRef(check);
705 ITfKeyEventSink_OnSetFocus(check, TRUE);
706 This->forgroundKeyEventSink = check;
707 This->forgroundTextService = textservice;
712 static HRESULT WINAPI KeystrokeMgr_UnadviseKeyEventSink(ITfKeystrokeMgr *iface,
715 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
717 ITfKeyEventSink *check = NULL;
718 TRACE("(%p) %x\n",This,tid);
723 textservice = get_textservice_clsid(tid);
724 if (IsEqualCLSID(&GUID_NULL,&textservice))
727 get_textservice_sink(tid, &IID_ITfKeyEventSink, (IUnknown**)&check);
730 return CONNECT_E_NOCONNECTION;
732 set_textservice_sink(tid, &IID_ITfKeyEventSink, NULL);
733 ITfKeyEventSink_Release(check);
735 if (This->forgroundKeyEventSink == check)
737 ITfKeyEventSink_Release(This->forgroundKeyEventSink);
738 This->forgroundKeyEventSink = NULL;
739 This->forgroundTextService = GUID_NULL;
744 static HRESULT WINAPI KeystrokeMgr_GetForeground(ITfKeystrokeMgr *iface,
747 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
748 TRACE("(%p) %p\n",This,pclsid);
752 if (IsEqualCLSID(&This->forgroundTextService,&GUID_NULL))
755 *pclsid = This->forgroundTextService;
759 static HRESULT WINAPI KeystrokeMgr_TestKeyDown(ITfKeystrokeMgr *iface,
760 WPARAM wParam, LPARAM lParam, BOOL *pfEaten)
762 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
763 FIXME("STUB:(%p)\n",This);
767 static HRESULT WINAPI KeystrokeMgr_TestKeyUp(ITfKeystrokeMgr *iface,
768 WPARAM wParam, LPARAM lParam, BOOL *pfEaten)
770 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
771 FIXME("STUB:(%p)\n",This);
775 static HRESULT WINAPI KeystrokeMgr_KeyDown(ITfKeystrokeMgr *iface,
776 WPARAM wParam, LPARAM lParam, BOOL *pfEaten)
778 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
779 FIXME("STUB:(%p)\n",This);
783 static HRESULT WINAPI KeystrokeMgr_KeyUp(ITfKeystrokeMgr *iface,
784 WPARAM wParam, LPARAM lParam, BOOL *pfEaten)
786 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
787 FIXME("STUB:(%p)\n",This);
791 static HRESULT WINAPI KeystrokeMgr_GetPreservedKey(ITfKeystrokeMgr *iface,
792 ITfContext *pic, const TF_PRESERVEDKEY *pprekey, GUID *pguid)
794 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
795 FIXME("STUB:(%p)\n",This);
799 static HRESULT WINAPI KeystrokeMgr_IsPreservedKey(ITfKeystrokeMgr *iface,
800 REFGUID rguid, const TF_PRESERVEDKEY *pprekey, BOOL *pfRegistered)
802 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
805 TRACE("(%p) %s (%x %x) %p\n",This,debugstr_guid(rguid), (pprekey)?pprekey->uVKey:0, (pprekey)?pprekey->uModifiers:0, pfRegistered);
807 if (!rguid || !pprekey || !pfRegistered)
810 LIST_FOR_EACH(cursor, &This->CurrentPreservedKeys)
812 PreservedKey* key = LIST_ENTRY(cursor,PreservedKey,entry);
813 if (IsEqualGUID(rguid,&key->guid) && pprekey->uVKey == key->prekey.uVKey && pprekey->uModifiers == key->prekey.uModifiers)
815 *pfRegistered = TRUE;
820 *pfRegistered = FALSE;
824 static HRESULT WINAPI KeystrokeMgr_PreserveKey(ITfKeystrokeMgr *iface,
825 TfClientId tid, REFGUID rguid, const TF_PRESERVEDKEY *prekey,
826 const WCHAR *pchDesc, ULONG cchDesc)
828 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
830 PreservedKey *newkey;
832 TRACE("(%p) %x %s (%x,%x) %s\n",This,tid, debugstr_guid(rguid),(prekey)?prekey->uVKey:0,(prekey)?prekey->uModifiers:0,debugstr_wn(pchDesc,cchDesc));
834 if (!tid || ! rguid || !prekey || (cchDesc && !pchDesc))
837 LIST_FOR_EACH(cursor, &This->CurrentPreservedKeys)
839 PreservedKey* key = LIST_ENTRY(cursor,PreservedKey,entry);
840 if (IsEqualGUID(rguid,&key->guid) && prekey->uVKey == key->prekey.uVKey && prekey->uModifiers == key->prekey.uModifiers)
841 return TF_E_ALREADY_EXISTS;
844 newkey = HeapAlloc(GetProcessHeap(),0,sizeof(PreservedKey));
846 return E_OUTOFMEMORY;
848 newkey->guid = *rguid;
849 newkey->prekey = *prekey;
853 newkey->description = HeapAlloc(GetProcessHeap(),0,cchDesc * sizeof(WCHAR));
854 if (!newkey->description)
856 HeapFree(GetProcessHeap(),0,newkey);
857 return E_OUTOFMEMORY;
859 memcpy(newkey->description, pchDesc, cchDesc*sizeof(WCHAR));
862 list_add_head(&This->CurrentPreservedKeys,&newkey->entry);
867 static HRESULT WINAPI KeystrokeMgr_UnpreserveKey(ITfKeystrokeMgr *iface,
868 REFGUID rguid, const TF_PRESERVEDKEY *pprekey)
870 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
871 PreservedKey* key = NULL;
873 TRACE("(%p) %s (%x %x)\n",This,debugstr_guid(rguid),(pprekey)?pprekey->uVKey:0, (pprekey)?pprekey->uModifiers:0);
875 if (!pprekey || !rguid)
878 LIST_FOR_EACH(cursor, &This->CurrentPreservedKeys)
880 key = LIST_ENTRY(cursor,PreservedKey,entry);
881 if (IsEqualGUID(rguid,&key->guid) && pprekey->uVKey == key->prekey.uVKey && pprekey->uModifiers == key->prekey.uModifiers)
887 return CONNECT_E_NOCONNECTION;
889 list_remove(&key->entry);
890 HeapFree(GetProcessHeap(),0,key->description);
891 HeapFree(GetProcessHeap(),0,key);
896 static HRESULT WINAPI KeystrokeMgr_SetPreservedKeyDescription(ITfKeystrokeMgr *iface,
897 REFGUID rguid, const WCHAR *pchDesc, ULONG cchDesc)
899 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
900 FIXME("STUB:(%p)\n",This);
904 static HRESULT WINAPI KeystrokeMgr_GetPreservedKeyDescription(ITfKeystrokeMgr *iface,
905 REFGUID rguid, BSTR *pbstrDesc)
907 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
908 FIXME("STUB:(%p)\n",This);
912 static HRESULT WINAPI KeystrokeMgr_SimulatePreservedKey(ITfKeystrokeMgr *iface,
913 ITfContext *pic, REFGUID rguid, BOOL *pfEaten)
915 ThreadMgr *This = impl_from_ITfKeystrokeMgrVtbl(iface);
916 FIXME("STUB:(%p)\n",This);
920 static const ITfKeystrokeMgrVtbl ThreadMgr_KeystrokeMgrVtbl =
922 KeystrokeMgr_QueryInterface,
924 KeystrokeMgr_Release,
926 KeystrokeMgr_AdviseKeyEventSink,
927 KeystrokeMgr_UnadviseKeyEventSink,
928 KeystrokeMgr_GetForeground,
929 KeystrokeMgr_TestKeyDown,
930 KeystrokeMgr_TestKeyUp,
931 KeystrokeMgr_KeyDown,
933 KeystrokeMgr_GetPreservedKey,
934 KeystrokeMgr_IsPreservedKey,
935 KeystrokeMgr_PreserveKey,
936 KeystrokeMgr_UnpreserveKey,
937 KeystrokeMgr_SetPreservedKeyDescription,
938 KeystrokeMgr_GetPreservedKeyDescription,
939 KeystrokeMgr_SimulatePreservedKey
942 /*****************************************************
943 * ITfMessagePump functions
944 *****************************************************/
946 static HRESULT WINAPI MessagePump_QueryInterface(ITfMessagePump *iface, REFIID iid, LPVOID *ppvOut)
948 ThreadMgr *This = impl_from_ITfMessagePumpVtbl(iface);
949 return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut);
952 static ULONG WINAPI MessagePump_AddRef(ITfMessagePump *iface)
954 ThreadMgr *This = impl_from_ITfMessagePumpVtbl(iface);
955 return ThreadMgr_AddRef((ITfThreadMgr*)This);
958 static ULONG WINAPI MessagePump_Release(ITfMessagePump *iface)
960 ThreadMgr *This = impl_from_ITfMessagePumpVtbl(iface);
961 return ThreadMgr_Release((ITfThreadMgr *)This);
964 static HRESULT WINAPI MessagePump_PeekMessageA(ITfMessagePump *iface,
965 LPMSG pMsg, HWND hwnd, UINT wMsgFilterMin, UINT wMsgFilterMax,
966 UINT wRemoveMsg, BOOL *pfResult)
970 *pfResult = PeekMessageA(pMsg, hwnd, wMsgFilterMin, wMsgFilterMax, wRemoveMsg);
974 static HRESULT WINAPI MessagePump_GetMessageA(ITfMessagePump *iface,
975 LPMSG pMsg, HWND hwnd, UINT wMsgFilterMin, UINT wMsgFilterMax,
980 *pfResult = GetMessageA(pMsg, hwnd, wMsgFilterMin, wMsgFilterMax);
984 static HRESULT WINAPI MessagePump_PeekMessageW(ITfMessagePump *iface,
985 LPMSG pMsg, HWND hwnd, UINT wMsgFilterMin, UINT wMsgFilterMax,
986 UINT wRemoveMsg, BOOL *pfResult)
990 *pfResult = PeekMessageW(pMsg, hwnd, wMsgFilterMin, wMsgFilterMax, wRemoveMsg);
994 static HRESULT WINAPI MessagePump_GetMessageW(ITfMessagePump *iface,
995 LPMSG pMsg, HWND hwnd, UINT wMsgFilterMin, UINT wMsgFilterMax,
1000 *pfResult = GetMessageW(pMsg, hwnd, wMsgFilterMin, wMsgFilterMax);
1004 static const ITfMessagePumpVtbl ThreadMgr_MessagePumpVtbl =
1006 MessagePump_QueryInterface,
1008 MessagePump_Release,
1010 MessagePump_PeekMessageA,
1011 MessagePump_GetMessageA,
1012 MessagePump_PeekMessageW,
1013 MessagePump_GetMessageW
1016 /*****************************************************
1017 * ITfClientId functions
1018 *****************************************************/
1020 static HRESULT WINAPI ClientId_QueryInterface(ITfClientId *iface, REFIID iid, LPVOID *ppvOut)
1022 ThreadMgr *This = impl_from_ITfClientIdVtbl(iface);
1023 return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut);
1026 static ULONG WINAPI ClientId_AddRef(ITfClientId *iface)
1028 ThreadMgr *This = impl_from_ITfClientIdVtbl(iface);
1029 return ThreadMgr_AddRef((ITfThreadMgr*)This);
1032 static ULONG WINAPI ClientId_Release(ITfClientId *iface)
1034 ThreadMgr *This = impl_from_ITfClientIdVtbl(iface);
1035 return ThreadMgr_Release((ITfThreadMgr *)This);
1038 static HRESULT WINAPI ClientId_GetClientId(ITfClientId *iface,
1039 REFCLSID rclsid, TfClientId *ptid)
1043 ITfCategoryMgr *catmgr;
1044 ThreadMgr *This = impl_from_ITfClientIdVtbl(iface);
1046 TRACE("(%p) %s\n",This,debugstr_guid(rclsid));
1048 CategoryMgr_Constructor(NULL,(IUnknown**)&catmgr);
1049 hr = ITfCategoryMgr_RegisterGUID(catmgr,rclsid,ptid);
1050 ITfCategoryMgr_Release(catmgr);
1055 static const ITfClientIdVtbl ThreadMgr_ClientIdVtbl =
1057 ClientId_QueryInterface,
1061 ClientId_GetClientId
1064 /*****************************************************
1065 * ITfThreadMgrEventSink functions (internal)
1066 *****************************************************/
1067 static HRESULT WINAPI ThreadMgrEventSink_QueryInterface(ITfThreadMgrEventSink *iface, REFIID iid, LPVOID *ppvOut)
1069 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
1070 return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut);
1073 static ULONG WINAPI ThreadMgrEventSink_AddRef(ITfThreadMgrEventSink *iface)
1075 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
1076 return ThreadMgr_AddRef((ITfThreadMgr*)This);
1079 static ULONG WINAPI ThreadMgrEventSink_Release(ITfThreadMgrEventSink *iface)
1081 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
1082 return ThreadMgr_Release((ITfThreadMgr *)This);
1086 static WINAPI HRESULT ThreadMgrEventSink_OnInitDocumentMgr(
1087 ITfThreadMgrEventSink *iface,ITfDocumentMgr *pdim)
1089 struct list *cursor;
1090 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
1092 TRACE("(%p) %p\n",This,pdim);
1094 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
1096 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
1097 ITfThreadMgrEventSink_OnInitDocumentMgr(sink->interfaces.pITfThreadMgrEventSink,pdim);
1103 static WINAPI HRESULT ThreadMgrEventSink_OnUninitDocumentMgr(
1104 ITfThreadMgrEventSink *iface, ITfDocumentMgr *pdim)
1106 struct list *cursor;
1107 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
1109 TRACE("(%p) %p\n",This,pdim);
1111 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
1113 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
1114 ITfThreadMgrEventSink_OnUninitDocumentMgr(sink->interfaces.pITfThreadMgrEventSink,pdim);
1120 static WINAPI HRESULT ThreadMgrEventSink_OnSetFocus(
1121 ITfThreadMgrEventSink *iface, ITfDocumentMgr *pdimFocus,
1122 ITfDocumentMgr *pdimPrevFocus)
1124 struct list *cursor;
1125 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
1127 TRACE("(%p) %p %p\n",This,pdimFocus, pdimPrevFocus);
1129 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
1131 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
1132 ITfThreadMgrEventSink_OnSetFocus(sink->interfaces.pITfThreadMgrEventSink, pdimFocus, pdimPrevFocus);
1138 static WINAPI HRESULT ThreadMgrEventSink_OnPushContext(
1139 ITfThreadMgrEventSink *iface, ITfContext *pic)
1141 struct list *cursor;
1142 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
1144 TRACE("(%p) %p\n",This,pic);
1146 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
1148 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
1149 ITfThreadMgrEventSink_OnPushContext(sink->interfaces.pITfThreadMgrEventSink,pic);
1155 static WINAPI HRESULT ThreadMgrEventSink_OnPopContext(
1156 ITfThreadMgrEventSink *iface, ITfContext *pic)
1158 struct list *cursor;
1159 ThreadMgr *This = impl_from_ITfThreadMgrEventSink(iface);
1161 TRACE("(%p) %p\n",This,pic);
1163 LIST_FOR_EACH(cursor, &This->ThreadMgrEventSink)
1165 ThreadMgrSink* sink = LIST_ENTRY(cursor,ThreadMgrSink,entry);
1166 ITfThreadMgrEventSink_OnPopContext(sink->interfaces.pITfThreadMgrEventSink,pic);
1172 static const ITfThreadMgrEventSinkVtbl ThreadMgr_ThreadMgrEventSinkVtbl =
1174 ThreadMgrEventSink_QueryInterface,
1175 ThreadMgrEventSink_AddRef,
1176 ThreadMgrEventSink_Release,
1178 ThreadMgrEventSink_OnInitDocumentMgr,
1179 ThreadMgrEventSink_OnUninitDocumentMgr,
1180 ThreadMgrEventSink_OnSetFocus,
1181 ThreadMgrEventSink_OnPushContext,
1182 ThreadMgrEventSink_OnPopContext
1185 /*****************************************************
1186 * ITfSourceSingle functions
1187 *****************************************************/
1188 static HRESULT WINAPI ThreadMgrSourceSingle_QueryInterface(ITfSourceSingle *iface, REFIID iid, LPVOID *ppvOut)
1190 ThreadMgr *This = impl_from_ITfSourceSingleVtbl(iface);
1191 return ThreadMgr_QueryInterface((ITfThreadMgr *)This, iid, *ppvOut);
1194 static ULONG WINAPI ThreadMgrSourceSingle_AddRef(ITfSourceSingle *iface)
1196 ThreadMgr *This = impl_from_ITfSourceSingleVtbl(iface);
1197 return ThreadMgr_AddRef((ITfThreadMgr *)This);
1200 static ULONG WINAPI ThreadMgrSourceSingle_Release(ITfSourceSingle *iface)
1202 ThreadMgr *This = impl_from_ITfSourceSingleVtbl(iface);
1203 return ThreadMgr_Release((ITfThreadMgr *)This);
1206 static WINAPI HRESULT ThreadMgrSourceSingle_AdviseSingleSink( ITfSourceSingle *iface,
1207 TfClientId tid, REFIID riid, IUnknown *punk)
1209 ThreadMgr *This = impl_from_ITfSourceSingleVtbl(iface);
1210 FIXME("STUB:(%p) %i %s %p\n",This, tid, debugstr_guid(riid),punk);
1214 static WINAPI HRESULT ThreadMgrSourceSingle_UnadviseSingleSink( ITfSourceSingle *iface,
1215 TfClientId tid, REFIID riid)
1217 ThreadMgr *This = impl_from_ITfSourceSingleVtbl(iface);
1218 FIXME("STUB:(%p) %i %s\n",This, tid, debugstr_guid(riid));
1222 static const ITfSourceSingleVtbl ThreadMgr_SourceSingleVtbl =
1224 ThreadMgrSourceSingle_QueryInterface,
1225 ThreadMgrSourceSingle_AddRef,
1226 ThreadMgrSourceSingle_Release,
1228 ThreadMgrSourceSingle_AdviseSingleSink,
1229 ThreadMgrSourceSingle_UnadviseSingleSink,
1232 HRESULT ThreadMgr_Constructor(IUnknown *pUnkOuter, IUnknown **ppOut)
1236 return CLASS_E_NOAGGREGATION;
1238 /* Only 1 ThreadMgr is created per thread */
1239 This = TlsGetValue(tlsIndex);
1242 ThreadMgr_AddRef((ITfThreadMgr*)This);
1243 *ppOut = (IUnknown*)This;
1247 This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(ThreadMgr));
1249 return E_OUTOFMEMORY;
1251 This->ThreadMgrVtbl= &ThreadMgr_ThreadMgrVtbl;
1252 This->SourceVtbl = &ThreadMgr_SourceVtbl;
1253 This->KeystrokeMgrVtbl= &ThreadMgr_KeystrokeMgrVtbl;
1254 This->MessagePumpVtbl= &ThreadMgr_MessagePumpVtbl;
1255 This->ClientIdVtbl = &ThreadMgr_ClientIdVtbl;
1256 This->ThreadMgrEventSinkVtbl = &ThreadMgr_ThreadMgrEventSinkVtbl;
1257 This->SourceSingleVtbl = &ThreadMgr_SourceSingleVtbl;
1259 TlsSetValue(tlsIndex,This);
1261 CompartmentMgr_Constructor((IUnknown*)This, &IID_IUnknown, (IUnknown**)&This->CompartmentMgr);
1263 list_init(&This->CurrentPreservedKeys);
1264 list_init(&This->CreatedDocumentMgrs);
1265 list_init(&This->AssociatedFocusWindows);
1267 list_init(&This->ActiveLanguageProfileNotifySink);
1268 list_init(&This->DisplayAttributeNotifySink);
1269 list_init(&This->KeyTraceEventSink);
1270 list_init(&This->PreservedKeyNotifySink);
1271 list_init(&This->ThreadFocusSink);
1272 list_init(&This->ThreadMgrEventSink);
1274 TRACE("returning %p\n", This);
1275 *ppOut = (IUnknown *)This;
1279 /**************************************************
1280 * IEnumTfDocumentMgrs implementaion
1281 **************************************************/
1282 static void EnumTfDocumentMgr_Destructor(EnumTfDocumentMgr *This)
1284 TRACE("destroying %p\n", This);
1285 HeapFree(GetProcessHeap(),0,This);
1288 static HRESULT WINAPI EnumTfDocumentMgr_QueryInterface(IEnumTfDocumentMgrs *iface, REFIID iid, LPVOID *ppvOut)
1290 EnumTfDocumentMgr *This = (EnumTfDocumentMgr *)iface;
1293 if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_IEnumTfDocumentMgrs))
1300 IUnknown_AddRef(iface);
1304 WARN("unsupported interface: %s\n", debugstr_guid(iid));
1305 return E_NOINTERFACE;
1308 static ULONG WINAPI EnumTfDocumentMgr_AddRef(IEnumTfDocumentMgrs *iface)
1310 EnumTfDocumentMgr *This = (EnumTfDocumentMgr*)iface;
1311 return InterlockedIncrement(&This->refCount);
1314 static ULONG WINAPI EnumTfDocumentMgr_Release(IEnumTfDocumentMgrs *iface)
1316 EnumTfDocumentMgr *This = (EnumTfDocumentMgr *)iface;
1319 ret = InterlockedDecrement(&This->refCount);
1321 EnumTfDocumentMgr_Destructor(This);
1325 static HRESULT WINAPI EnumTfDocumentMgr_Next(IEnumTfDocumentMgrs *iface,
1326 ULONG ulCount, ITfDocumentMgr **rgDocumentMgr, ULONG *pcFetched)
1328 EnumTfDocumentMgr *This = (EnumTfDocumentMgr *)iface;
1331 TRACE("(%p)\n",This);
1333 if (rgDocumentMgr == NULL) return E_POINTER;
1335 while (fetched < ulCount)
1337 DocumentMgrEntry *mgrentry;
1338 if (This->index == NULL)
1341 mgrentry = LIST_ENTRY(This->index,DocumentMgrEntry,entry);
1342 if (mgrentry == NULL)
1345 *rgDocumentMgr = mgrentry->docmgr;
1346 ITfDocumentMgr_AddRef(*rgDocumentMgr);
1348 This->index = list_next(This->head, This->index);
1353 if (pcFetched) *pcFetched = fetched;
1354 return fetched == ulCount ? S_OK : S_FALSE;
1357 static HRESULT WINAPI EnumTfDocumentMgr_Skip( IEnumTfDocumentMgrs* iface, ULONG celt)
1360 EnumTfDocumentMgr *This = (EnumTfDocumentMgr *)iface;
1361 TRACE("(%p)\n",This);
1362 for(i = 0; i < celt && This->index != NULL; i++)
1363 This->index = list_next(This->head, This->index);
1367 static HRESULT WINAPI EnumTfDocumentMgr_Reset( IEnumTfDocumentMgrs* iface)
1369 EnumTfDocumentMgr *This = (EnumTfDocumentMgr *)iface;
1370 TRACE("(%p)\n",This);
1371 This->index = list_head(This->head);
1375 static HRESULT WINAPI EnumTfDocumentMgr_Clone( IEnumTfDocumentMgrs *iface,
1376 IEnumTfDocumentMgrs **ppenum)
1378 EnumTfDocumentMgr *This = (EnumTfDocumentMgr *)iface;
1381 TRACE("(%p)\n",This);
1383 if (ppenum == NULL) return E_POINTER;
1385 res = EnumTfDocumentMgr_Constructor(This->head, ppenum);
1388 EnumTfDocumentMgr *new_This = (EnumTfDocumentMgr *)*ppenum;
1389 new_This->index = This->index;
1394 static const IEnumTfDocumentMgrsVtbl IEnumTfDocumentMgrs_Vtbl ={
1395 EnumTfDocumentMgr_QueryInterface,
1396 EnumTfDocumentMgr_AddRef,
1397 EnumTfDocumentMgr_Release,
1399 EnumTfDocumentMgr_Clone,
1400 EnumTfDocumentMgr_Next,
1401 EnumTfDocumentMgr_Reset,
1402 EnumTfDocumentMgr_Skip
1405 static HRESULT EnumTfDocumentMgr_Constructor(struct list* head, IEnumTfDocumentMgrs **ppOut)
1407 EnumTfDocumentMgr *This;
1409 This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(EnumTfDocumentMgr));
1411 return E_OUTOFMEMORY;
1413 This->Vtbl= &IEnumTfDocumentMgrs_Vtbl;
1416 This->index = list_head(This->head);
1418 TRACE("returning %p\n", This);
1419 *ppOut = (IEnumTfDocumentMgrs*)This;
1423 void ThreadMgr_OnDocumentMgrDestruction(ITfThreadMgr *tm, ITfDocumentMgr *mgr)
1425 ThreadMgr *This = (ThreadMgr *)tm;
1426 struct list *cursor;
1427 LIST_FOR_EACH(cursor, &This->CreatedDocumentMgrs)
1429 DocumentMgrEntry *mgrentry = LIST_ENTRY(cursor,DocumentMgrEntry,entry);
1430 if (mgrentry->docmgr == mgr)
1432 list_remove(cursor);
1433 HeapFree(GetProcessHeap(),0,mgrentry);
1437 FIXME("ITfDocumenMgr %p not found in this thread\n",mgr);
1440 LRESULT CALLBACK ThreadFocusHookProc(int nCode, WPARAM wParam, LPARAM lParam)
1444 This = TlsGetValue(tlsIndex);
1447 ERR("Hook proc but no ThreadMgr for this thread. Serious Error\n");
1450 if (!This->focusHook)
1452 ERR("Hook proc but no ThreadMgr focus Hook. Serious Error\n");
1456 if (nCode == HCBT_SETFOCUS) /* focus change within our thread */
1458 struct list *cursor;
1460 LIST_FOR_EACH(cursor, &This->AssociatedFocusWindows)
1462 AssociatedWindow *wnd = LIST_ENTRY(cursor,AssociatedWindow,entry);
1463 if (wnd->hwnd == (HWND)wParam)
1465 TRACE("Triggering Associated window focus\n");
1466 if (This->focus != wnd->docmgr)
1467 ThreadMgr_SetFocus((ITfThreadMgr*)This, wnd->docmgr);
1473 return CallNextHookEx(This->focusHook, nCode, wParam, lParam);