2 * Unit tests for ITfInputProcessor
4 * Copyright 2009 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
24 #include "wine/test.h"
32 static ITfInputProcessorProfiles
* g_ipp
;
33 static LANGID gLangid
;
34 static ITfCategoryMgr
* g_cm
= NULL
;
35 static ITfThreadMgr
* g_tm
= NULL
;
36 static ITfDocumentMgr
*g_dm
= NULL
;
37 static TfClientId cid
= 0;
38 static TfClientId tid
= 0;
40 #define SINK_UNEXPECTED 0
41 #define SINK_EXPECTED 1
44 static BOOL test_ShouldActivate
= FALSE
;
45 static BOOL test_ShouldDeactivate
= FALSE
;
47 static DWORD tmSinkCookie
;
48 static DWORD tmSinkRefCount
;
49 static ITfDocumentMgr
*test_CurrentFocus
= NULL
;
50 static ITfDocumentMgr
*test_PrevFocus
= NULL
;
51 static INT test_OnSetFocus
= SINK_UNEXPECTED
;
53 HRESULT
RegisterTextService(REFCLSID rclsid
);
54 HRESULT
UnregisterTextService();
55 HRESULT
ThreadMgrEventSink_Constructor(IUnknown
**ppOut
);
57 DEFINE_GUID(CLSID_FakeService
, 0xEDE1A7AD,0x66DE,0x47E0,0xB6,0x20,0x3E,0x92,0xF8,0x24,0x6B,0xF3);
58 DEFINE_GUID(CLSID_TF_InputProcessorProfiles
, 0x33c53a50,0xf456,0x4884,0xb0,0x49,0x85,0xfd,0x64,0x3e,0xcf,0xed);
59 DEFINE_GUID(CLSID_TF_CategoryMgr
, 0xA4B544A1,0x438D,0x4B41,0x93,0x25,0x86,0x95,0x23,0xE2,0xD6,0xC7);
60 DEFINE_GUID(GUID_TFCAT_TIP_KEYBOARD
, 0x34745c63,0xb2f0,0x4784,0x8b,0x67,0x5e,0x12,0xc8,0x70,0x1a,0x31);
61 DEFINE_GUID(GUID_TFCAT_TIP_SPEECH
, 0xB5A73CD1,0x8355,0x426B,0xA1,0x61,0x25,0x98,0x08,0xF2,0x6B,0x14);
62 DEFINE_GUID(GUID_TFCAT_TIP_HANDWRITING
, 0x246ecb87,0xc2f2,0x4abe,0x90,0x5b,0xc8,0xb3,0x8a,0xdd,0x2c,0x43);
63 DEFINE_GUID (GUID_TFCAT_DISPLAYATTRIBUTEPROVIDER
, 0x046B8C80,0x1647,0x40F7,0x9B,0x21,0xB9,0x3B,0x81,0xAA,0xBC,0x1B);
64 DEFINE_GUID(GUID_NULL
,0,0,0,0,0,0,0,0,0,0,0);
65 DEFINE_GUID(CLSID_TF_ThreadMgr
, 0x529a9e6b,0x6587,0x4f23,0xab,0x9e,0x9c,0x7d,0x68,0x3e,0x3c,0x50);
68 static HRESULT
initialize(void)
72 hr
= CoCreateInstance (&CLSID_TF_InputProcessorProfiles
, NULL
,
73 CLSCTX_INPROC_SERVER
, &IID_ITfInputProcessorProfiles
, (void**)&g_ipp
);
75 hr
= CoCreateInstance (&CLSID_TF_CategoryMgr
, NULL
,
76 CLSCTX_INPROC_SERVER
, &IID_ITfCategoryMgr
, (void**)&g_cm
);
78 hr
= CoCreateInstance (&CLSID_TF_ThreadMgr
, NULL
,
79 CLSCTX_INPROC_SERVER
, &IID_ITfThreadMgr
, (void**)&g_tm
);
83 static void cleanup(void)
86 ITfInputProcessorProfiles_Release(g_ipp
);
88 ITfCategoryMgr_Release(g_cm
);
90 ITfThreadMgr_Release(g_tm
);
94 static void test_Register(void)
98 static const WCHAR szDesc
[] = {'F','a','k','e',' ','W','i','n','e',' ','S','e','r','v','i','c','e',0};
99 static const WCHAR szFile
[] = {'F','a','k','e',' ','W','i','n','e',' ','S','e','r','v','i','c','e',' ','F','i','l','e',0};
101 hr
= RegisterTextService(&CLSID_FakeService
);
102 ok(SUCCEEDED(hr
),"Unable to register COM for TextService\n");
103 hr
= ITfInputProcessorProfiles_Register(g_ipp
, &CLSID_FakeService
);
104 ok(SUCCEEDED(hr
),"Unable to register text service(%x)\n",hr
);
105 hr
= ITfInputProcessorProfiles_AddLanguageProfile(g_ipp
, &CLSID_FakeService
, gLangid
, &CLSID_FakeService
, szDesc
, sizeof(szDesc
)/sizeof(WCHAR
), szFile
, sizeof(szFile
)/sizeof(WCHAR
), 1);
106 ok(SUCCEEDED(hr
),"Unable to add Language Profile (%x)\n",hr
);
109 static void test_Unregister(void)
112 hr
= ITfInputProcessorProfiles_Unregister(g_ipp
, &CLSID_FakeService
);
113 ok(SUCCEEDED(hr
),"Unable to unregister text service(%x)\n",hr
);
114 UnregisterTextService();
117 static void test_EnumInputProcessorInfo(void)
122 if (SUCCEEDED(ITfInputProcessorProfiles_EnumInputProcessorInfo(g_ipp
, &ppEnum
)))
126 while (IEnumGUID_Next(ppEnum
, 1, &g
, &fetched
) == S_OK
)
128 if(IsEqualGUID(&g
,&CLSID_FakeService
))
132 ok(found
,"Did not find registered text service\n");
135 static void test_EnumLanguageProfiles(void)
138 IEnumTfLanguageProfiles
*ppEnum
;
139 if (SUCCEEDED(ITfInputProcessorProfiles_EnumLanguageProfiles(g_ipp
,gLangid
,&ppEnum
)))
141 TF_LANGUAGEPROFILE profile
;
142 while (IEnumTfLanguageProfiles_Next(ppEnum
,1,&profile
,NULL
)==S_OK
)
144 if (IsEqualGUID(&profile
.clsid
,&CLSID_FakeService
))
147 ok(profile
.langid
== gLangid
, "LangId Incorrect\n");
148 ok(IsEqualGUID(&profile
.catid
,&GUID_TFCAT_TIP_KEYBOARD
), "CatId Incorrect\n");
149 ok(IsEqualGUID(&profile
.guidProfile
,&CLSID_FakeService
), "guidProfile Incorrect\n");
153 ok(found
,"Registered text service not found\n");
156 static void test_RegisterCategory(void)
159 hr
= ITfCategoryMgr_RegisterCategory(g_cm
, &CLSID_FakeService
, &GUID_TFCAT_TIP_KEYBOARD
, &CLSID_FakeService
);
160 ok(SUCCEEDED(hr
),"ITfCategoryMgr_RegisterCategory failed\n");
161 hr
= ITfCategoryMgr_RegisterCategory(g_cm
, &CLSID_FakeService
, &GUID_TFCAT_DISPLAYATTRIBUTEPROVIDER
, &CLSID_FakeService
);
162 ok(SUCCEEDED(hr
),"ITfCategoryMgr_RegisterCategory failed\n");
165 static void test_UnregisterCategory(void)
168 hr
= ITfCategoryMgr_UnregisterCategory(g_cm
, &CLSID_FakeService
, &GUID_TFCAT_TIP_KEYBOARD
, &CLSID_FakeService
);
169 todo_wine
ok(SUCCEEDED(hr
),"ITfCategoryMgr_UnregisterCategory failed\n");
170 hr
= ITfCategoryMgr_UnregisterCategory(g_cm
, &CLSID_FakeService
, &GUID_TFCAT_DISPLAYATTRIBUTEPROVIDER
, &CLSID_FakeService
);
171 todo_wine
ok(SUCCEEDED(hr
),"ITfCategoryMgr_UnregisterCategory failed\n");
174 static void test_FindClosestCategory(void)
178 const GUID
*list
[3] = {&GUID_TFCAT_TIP_SPEECH
, &GUID_TFCAT_TIP_KEYBOARD
, &GUID_TFCAT_TIP_HANDWRITING
};
180 hr
= ITfCategoryMgr_FindClosestCategory(g_cm
, &CLSID_FakeService
, &output
, NULL
, 0);
181 ok(SUCCEEDED(hr
),"ITfCategoryMgr_FindClosestCategory failed (%x)\n",hr
);
182 ok(IsEqualGUID(&output
,&GUID_TFCAT_DISPLAYATTRIBUTEPROVIDER
),"Wrong GUID\n");
184 hr
= ITfCategoryMgr_FindClosestCategory(g_cm
, &CLSID_FakeService
, &output
, list
, 1);
185 ok(SUCCEEDED(hr
),"ITfCategoryMgr_FindClosestCategory failed (%x)\n",hr
);
186 ok(IsEqualGUID(&output
,&GUID_NULL
),"Wrong GUID\n");
188 hr
= ITfCategoryMgr_FindClosestCategory(g_cm
, &CLSID_FakeService
, &output
, list
, 3);
189 ok(SUCCEEDED(hr
),"ITfCategoryMgr_FindClosestCategory failed (%x)\n",hr
);
190 ok(IsEqualGUID(&output
,&GUID_TFCAT_TIP_KEYBOARD
),"Wrong GUID\n");
193 static void test_Enable(void)
196 BOOL enabled
= FALSE
;
198 hr
= ITfInputProcessorProfiles_EnableLanguageProfile(g_ipp
,&CLSID_FakeService
, gLangid
, &CLSID_FakeService
, TRUE
);
199 ok(SUCCEEDED(hr
),"Failed to enable text service\n");
200 hr
= ITfInputProcessorProfiles_IsEnabledLanguageProfile(g_ipp
,&CLSID_FakeService
, gLangid
, &CLSID_FakeService
, &enabled
);
201 ok(SUCCEEDED(hr
),"Failed to get enabled state\n");
202 ok(enabled
== TRUE
,"enabled state incorrect\n");
205 static void test_Disable(void)
209 trace("Disabling\n");
210 hr
= ITfInputProcessorProfiles_EnableLanguageProfile(g_ipp
,&CLSID_FakeService
, gLangid
, &CLSID_FakeService
, FALSE
);
211 ok(SUCCEEDED(hr
),"Failed to disable text service\n");
214 static void test_ThreadMgrAdviseSinks(void)
216 ITfSource
*source
= NULL
;
220 hr
= ITfThreadMgr_QueryInterface(g_tm
, &IID_ITfSource
, (LPVOID
*)&source
);
221 ok(SUCCEEDED(hr
),"Failed to get IID_ITfSource for ThreadMgr\n");
225 ThreadMgrEventSink_Constructor(&sink
);
229 hr
= ITfSource_AdviseSink(source
,&IID_ITfThreadMgrEventSink
, sink
, &tmSinkCookie
);
230 ok(SUCCEEDED(hr
),"Failed to Advise Sink\n");
231 ok(tmSinkCookie
!=0,"Failed to get sink cookie\n");
233 /* Advising the sink adds a ref, Relesing here lets the object be deleted
236 IUnknown_Release(sink
);
237 ITfSource_Release(source
);
240 static void test_ThreadMgrUnadviseSinks(void)
242 ITfSource
*source
= NULL
;
245 hr
= ITfThreadMgr_QueryInterface(g_tm
, &IID_ITfSource
, (LPVOID
*)&source
);
246 ok(SUCCEEDED(hr
),"Failed to get IID_ITfSource for ThreadMgr\n");
251 hr
= ITfSource_UnadviseSink(source
, tmSinkCookie
);
252 ok(SUCCEEDED(hr
),"Failed to unadvise Sink\n");
253 ITfSource_Release(source
);
256 static void test_Activate(void)
260 hr
= ITfInputProcessorProfiles_ActivateLanguageProfile(g_ipp
,&CLSID_FakeService
,gLangid
,&CLSID_FakeService
);
261 todo_wine
ok(SUCCEEDED(hr
),"Failed to Activate text service\n");
264 static void test_startSession(void)
268 ITfDocumentMgr
*dmtest
;
270 test_ShouldActivate
= TRUE
;
271 ITfThreadMgr_Activate(g_tm
,&cid
);
272 todo_wine
ok(cid
!= tid
,"TextService id mistakenly matches Client id\n");
274 hr
= ITfThreadMgr_CreateDocumentMgr(g_tm
,&g_dm
);
275 ok(SUCCEEDED(hr
),"CreateDocumentMgr failed\n");
277 hr
= ITfThreadMgr_GetFocus(g_tm
,&dmtest
);
278 ok(SUCCEEDED(hr
),"GetFocus Failed\n");
279 ok(dmtest
== NULL
,"Initial focus not null\n");
281 test_CurrentFocus
= g_dm
;
282 test_PrevFocus
= NULL
;
283 test_OnSetFocus
= SINK_EXPECTED
;
284 hr
= ITfThreadMgr_SetFocus(g_tm
,g_dm
);
285 ok(SUCCEEDED(hr
),"SetFocus Failed\n");
286 ok(test_OnSetFocus
== SINK_FIRED
, "OnSetFocus sink not called\n");
287 test_OnSetFocus
= SINK_UNEXPECTED
;
289 hr
= ITfThreadMgr_GetFocus(g_tm
,&dmtest
);
290 ok(SUCCEEDED(hr
),"GetFocus Failed\n");
291 ok(g_dm
== dmtest
,"Expected DocumentMgr not focused\n");
293 cnt
= ITfDocumentMgr_Release(g_dm
);
294 ok(cnt
== 2,"DocumentMgr refcount not expected (2 vs %i)\n",cnt
);
296 hr
= ITfThreadMgr_GetFocus(g_tm
,&dmtest
);
297 ok(SUCCEEDED(hr
),"GetFocus Failed\n");
298 ok(g_dm
== dmtest
,"Expected DocumentMgr not focused\n");
301 static void test_endSession(void)
303 test_ShouldDeactivate
= TRUE
;
304 test_CurrentFocus
= NULL
;
305 test_PrevFocus
= g_dm
;
306 test_OnSetFocus
= SINK_EXPECTED
;
307 ITfThreadMgr_Deactivate(g_tm
);
308 ok(test_OnSetFocus
== SINK_FIRED
, "OnSetFocus sink not called\n");
309 test_OnSetFocus
= SINK_UNEXPECTED
;
312 START_TEST(inputprocessor
)
314 if (SUCCEEDED(initialize()))
316 gLangid
= GetUserDefaultLCID();
318 test_RegisterCategory();
319 test_EnumInputProcessorInfo();
321 test_ThreadMgrAdviseSinks();
325 test_EnumLanguageProfiles();
326 test_FindClosestCategory();
328 test_ThreadMgrUnadviseSinks();
329 test_UnregisterCategory();
333 skip("Unable to create InputProcessor\n");
337 /**********************************************************************
338 * ITfThreadMgrEventSink
339 **********************************************************************/
340 typedef struct tagThreadMgrEventSink
342 const ITfThreadMgrEventSinkVtbl
*ThreadMgrEventSinkVtbl
;
344 } ThreadMgrEventSink
;
346 static void ThreadMgrEventSink_Destructor(ThreadMgrEventSink
*This
)
348 HeapFree(GetProcessHeap(),0,This
);
351 static HRESULT WINAPI
ThreadMgrEventSink_QueryInterface(ITfThreadMgrEventSink
*iface
, REFIID iid
, LPVOID
*ppvOut
)
353 ThreadMgrEventSink
*This
= (ThreadMgrEventSink
*)iface
;
356 if (IsEqualIID(iid
, &IID_IUnknown
) || IsEqualIID(iid
, &IID_ITfThreadMgrEventSink
))
363 IUnknown_AddRef(iface
);
367 return E_NOINTERFACE
;
370 static ULONG WINAPI
ThreadMgrEventSink_AddRef(ITfThreadMgrEventSink
*iface
)
372 ThreadMgrEventSink
*This
= (ThreadMgrEventSink
*)iface
;
373 ok (tmSinkRefCount
== This
->refCount
,"ThreadMgrEventSink refcount off %i vs %i\n",This
->refCount
,tmSinkRefCount
);
374 return InterlockedIncrement(&This
->refCount
);
377 static ULONG WINAPI
ThreadMgrEventSink_Release(ITfThreadMgrEventSink
*iface
)
379 ThreadMgrEventSink
*This
= (ThreadMgrEventSink
*)iface
;
382 ok (tmSinkRefCount
== This
->refCount
,"ThreadMgrEventSink refcount off %i vs %i\n",This
->refCount
,tmSinkRefCount
);
383 ret
= InterlockedDecrement(&This
->refCount
);
385 ThreadMgrEventSink_Destructor(This
);
389 static HRESULT WINAPI
ThreadMgrEventSink_OnInitDocumentMgr(ITfThreadMgrEventSink
*iface
,
390 ITfDocumentMgr
*pdim
)
396 static HRESULT WINAPI
ThreadMgrEventSink_OnUninitDocumentMgr(ITfThreadMgrEventSink
*iface
,
397 ITfDocumentMgr
*pdim
)
403 static HRESULT WINAPI
ThreadMgrEventSink_OnSetFocus(ITfThreadMgrEventSink
*iface
,
404 ITfDocumentMgr
*pdimFocus
, ITfDocumentMgr
*pdimPrevFocus
)
406 ok(test_OnSetFocus
== SINK_EXPECTED
, "Unexpected OnSetFocus sink\n");
407 ok(pdimFocus
== test_CurrentFocus
,"Sink reports wrong focus\n");
408 ok(pdimPrevFocus
== test_PrevFocus
,"Sink reports wrong previous focus\n");
409 test_OnSetFocus
= SINK_FIRED
;
413 static HRESULT WINAPI
ThreadMgrEventSink_OnPushContext(ITfThreadMgrEventSink
*iface
,
420 static HRESULT WINAPI
ThreadMgrEventSink_OnPopContext(ITfThreadMgrEventSink
*iface
,
427 static const ITfThreadMgrEventSinkVtbl ThreadMgrEventSink_ThreadMgrEventSinkVtbl
=
429 ThreadMgrEventSink_QueryInterface
,
430 ThreadMgrEventSink_AddRef
,
431 ThreadMgrEventSink_Release
,
433 ThreadMgrEventSink_OnInitDocumentMgr
,
434 ThreadMgrEventSink_OnUninitDocumentMgr
,
435 ThreadMgrEventSink_OnSetFocus
,
436 ThreadMgrEventSink_OnPushContext
,
437 ThreadMgrEventSink_OnPopContext
440 HRESULT
ThreadMgrEventSink_Constructor(IUnknown
**ppOut
)
442 ThreadMgrEventSink
*This
;
444 This
= HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY
,sizeof(ThreadMgrEventSink
));
446 return E_OUTOFMEMORY
;
448 This
->ThreadMgrEventSinkVtbl
= &ThreadMgrEventSink_ThreadMgrEventSinkVtbl
;
451 *ppOut
= (IUnknown
*)This
;
456 /********************************************************************************************
457 * Stub text service for testing
458 ********************************************************************************************/
460 static LONG TS_refCount
;
461 static IClassFactory
*cf
;
464 typedef HRESULT (*LPFNCONSTRUCTOR
)(IUnknown
*pUnkOuter
, IUnknown
**ppvOut
);
466 typedef struct tagClassFactory
468 const IClassFactoryVtbl
*vtbl
;
470 LPFNCONSTRUCTOR ctor
;
473 typedef struct tagTextService
475 const ITfTextInputProcessorVtbl
*TextInputProcessorVtbl
;
479 static void ClassFactory_Destructor(ClassFactory
*This
)
481 HeapFree(GetProcessHeap(),0,This
);
485 static HRESULT WINAPI
ClassFactory_QueryInterface(IClassFactory
*iface
, REFIID riid
, LPVOID
*ppvOut
)
488 if (IsEqualIID(riid
, &IID_IClassFactory
) || IsEqualIID(riid
, &IID_IUnknown
))
490 IClassFactory_AddRef(iface
);
495 return E_NOINTERFACE
;
498 static ULONG WINAPI
ClassFactory_AddRef(IClassFactory
*iface
)
500 ClassFactory
*This
= (ClassFactory
*)iface
;
501 return InterlockedIncrement(&This
->ref
);
504 static ULONG WINAPI
ClassFactory_Release(IClassFactory
*iface
)
506 ClassFactory
*This
= (ClassFactory
*)iface
;
507 ULONG ret
= InterlockedDecrement(&This
->ref
);
510 ClassFactory_Destructor(This
);
514 static HRESULT WINAPI
ClassFactory_CreateInstance(IClassFactory
*iface
, IUnknown
*punkOuter
, REFIID iid
, LPVOID
*ppvOut
)
516 ClassFactory
*This
= (ClassFactory
*)iface
;
520 ret
= This
->ctor(punkOuter
, &obj
);
523 ret
= IUnknown_QueryInterface(obj
, iid
, ppvOut
);
524 IUnknown_Release(obj
);
528 static HRESULT WINAPI
ClassFactory_LockServer(IClassFactory
*iface
, BOOL fLock
)
531 InterlockedIncrement(&TS_refCount
);
533 InterlockedDecrement(&TS_refCount
);
538 static const IClassFactoryVtbl ClassFactoryVtbl
= {
540 ClassFactory_QueryInterface
,
542 ClassFactory_Release
,
545 ClassFactory_CreateInstance
,
546 ClassFactory_LockServer
549 static HRESULT
ClassFactory_Constructor(LPFNCONSTRUCTOR ctor
, LPVOID
*ppvOut
)
551 ClassFactory
*This
= HeapAlloc(GetProcessHeap(),0,sizeof(ClassFactory
));
552 This
->vtbl
= &ClassFactoryVtbl
;
555 *ppvOut
= (LPVOID
)This
;
560 static void TextService_Destructor(TextService
*This
)
562 HeapFree(GetProcessHeap(),0,This
);
565 static HRESULT WINAPI
TextService_QueryInterface(ITfTextInputProcessor
*iface
, REFIID iid
, LPVOID
*ppvOut
)
567 TextService
*This
= (TextService
*)iface
;
570 if (IsEqualIID(iid
, &IID_IUnknown
) || IsEqualIID(iid
, &IID_ITfTextInputProcessor
))
577 IUnknown_AddRef(iface
);
581 return E_NOINTERFACE
;
584 static ULONG WINAPI
TextService_AddRef(ITfTextInputProcessor
*iface
)
586 TextService
*This
= (TextService
*)iface
;
587 return InterlockedIncrement(&This
->refCount
);
590 static ULONG WINAPI
TextService_Release(ITfTextInputProcessor
*iface
)
592 TextService
*This
= (TextService
*)iface
;
595 ret
= InterlockedDecrement(&This
->refCount
);
597 TextService_Destructor(This
);
601 static HRESULT WINAPI
TextService_Activate(ITfTextInputProcessor
*iface
,
602 ITfThreadMgr
*ptim
, TfClientId id
)
604 trace("TextService_Activate\n");
605 ok(test_ShouldActivate
,"Activation came unexpectedly\n");
610 static HRESULT WINAPI
TextService_Deactivate(ITfTextInputProcessor
*iface
)
612 trace("TextService_Deactivate\n");
613 ok(test_ShouldDeactivate
,"Deactivation came unexpectedly\n");
617 static const ITfTextInputProcessorVtbl TextService_TextInputProcessorVtbl
=
619 TextService_QueryInterface
,
623 TextService_Activate
,
624 TextService_Deactivate
627 HRESULT
TextService_Constructor(IUnknown
*pUnkOuter
, IUnknown
**ppOut
)
631 return CLASS_E_NOAGGREGATION
;
633 This
= HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY
,sizeof(TextService
));
635 return E_OUTOFMEMORY
;
637 This
->TextInputProcessorVtbl
= &TextService_TextInputProcessorVtbl
;
640 *ppOut
= (IUnknown
*)This
;
644 HRESULT
RegisterTextService(REFCLSID rclsid
)
646 ClassFactory_Constructor( TextService_Constructor
,(LPVOID
*)&cf
);
647 return CoRegisterClassObject(rclsid
, (IUnknown
*) cf
, CLSCTX_INPROC_SERVER
, REGCLS_MULTIPLEUSE
, ®id
);
650 HRESULT
UnregisterTextService()
652 return CoRevokeClassObject(regid
);