2 * ITfInputProcessorProfiles implementation
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
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 static const WCHAR szwLngp[] = {'L','a','n','g','u','a','g','e','P','r','o','f','i','l','e',0};
46 static const WCHAR szwEnable[] = {'E','n','a','b','l','e',0};
47 static const WCHAR szwTipfmt[] = {'%','s','\\','%','s',0};
48 static const WCHAR szwFullLangfmt[] = {'%','s','\\','%','s','\\','%','s','\\','0','x','%','0','8','x','\\','%','s',0};
50 static const WCHAR szwAssemblies[] = {'A','s','s','e','m','b','l','i','e','s',0};
51 static const WCHAR szwDefault[] = {'D','e','f','a','u','l','t',0};
52 static const WCHAR szwProfile[] = {'P','r','o','f','i','l','e',0};
53 static const WCHAR szwDefaultFmt[] = {'%','s','\\','%','s','\\','0','x','%','0','8','x','\\','%','s',0};
55 typedef struct tagInputProcessorProfilesSink {
58 /* InputProcessorProfile Sinks */
60 ITfLanguageProfileNotifySink *pITfLanguageProfileNotifySink;
62 } InputProcessorProfilesSink;
64 typedef struct tagInputProcessorProfiles {
65 const ITfInputProcessorProfilesVtbl *InputProcessorProfilesVtbl;
66 const ITfSourceVtbl *SourceVtbl;
67 /* const ITfInputProcessorProfileMgrVtbl *InputProcessorProfileMgrVtbl; */
68 /* const ITfInputProcessorProfilesExVtbl *InputProcessorProfilesExVtbl; */
69 /* const ITfInputProcessorProfileSubstituteLayoutVtbl *InputProcessorProfileSubstituteLayoutVtbl; */
72 LANGID currentLanguage;
74 struct list LanguageProfileNotifySink;
75 } InputProcessorProfiles;
77 typedef struct tagProfilesEnumGuid {
78 const IEnumGUIDVtbl *Vtbl;
85 typedef struct tagEnumTfLanguageProfiles {
86 const IEnumTfLanguageProfilesVtbl *Vtbl;
91 WCHAR szwCurrentClsid[39];
97 ITfCategoryMgr *catmgr;
98 } EnumTfLanguageProfiles;
100 static HRESULT ProfilesEnumGuid_Constructor(IEnumGUID **ppOut);
101 static HRESULT EnumTfLanguageProfiles_Constructor(LANGID langid, IEnumTfLanguageProfiles **ppOut);
103 static inline InputProcessorProfiles *impl_from_ITfSourceVtbl(ITfSource *iface)
105 return (InputProcessorProfiles *)((char *)iface - FIELD_OFFSET(InputProcessorProfiles,SourceVtbl));
108 static void free_sink(InputProcessorProfilesSink *sink)
110 IUnknown_Release(sink->interfaces.pIUnknown);
111 HeapFree(GetProcessHeap(),0,sink);
114 static void InputProcessorProfiles_Destructor(InputProcessorProfiles *This)
116 struct list *cursor, *cursor2;
117 TRACE("destroying %p\n", This);
120 LIST_FOR_EACH_SAFE(cursor, cursor2, &This->LanguageProfileNotifySink)
122 InputProcessorProfilesSink* sink = LIST_ENTRY(cursor,InputProcessorProfilesSink,entry);
127 HeapFree(GetProcessHeap(),0,This);
130 static void add_userkey( REFCLSID rclsid, LANGID langid,
137 DWORD disposition = 0;
142 StringFromGUID2(rclsid, buf, 39);
143 StringFromGUID2(guidProfile, buf2, 39);
144 sprintfW(fullkey,szwFullLangfmt,szwSystemTIPKey,buf,szwLngp,langid,buf2);
146 res = RegCreateKeyExW(HKEY_CURRENT_USER,fullkey, 0, NULL, 0,
147 KEY_READ | KEY_WRITE, NULL, &key, &disposition);
149 if (!res && disposition == REG_CREATED_NEW_KEY)
152 RegSetValueExW(key, szwEnable, 0, REG_DWORD, (LPBYTE)&zero, sizeof(DWORD));
159 static HRESULT WINAPI InputProcessorProfiles_QueryInterface(ITfInputProcessorProfiles *iface, REFIID iid, LPVOID *ppvOut)
161 InputProcessorProfiles *This = (InputProcessorProfiles *)iface;
164 if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_ITfInputProcessorProfiles))
168 else if (IsEqualIID(iid, &IID_ITfSource))
170 *ppvOut = &This->SourceVtbl;
175 IUnknown_AddRef(iface);
179 WARN("unsupported interface: %s\n", debugstr_guid(iid));
180 return E_NOINTERFACE;
183 static ULONG WINAPI InputProcessorProfiles_AddRef(ITfInputProcessorProfiles *iface)
185 InputProcessorProfiles *This = (InputProcessorProfiles *)iface;
186 return InterlockedIncrement(&This->refCount);
189 static ULONG WINAPI InputProcessorProfiles_Release(ITfInputProcessorProfiles *iface)
191 InputProcessorProfiles *This = (InputProcessorProfiles *)iface;
194 ret = InterlockedDecrement(&This->refCount);
196 InputProcessorProfiles_Destructor(This);
200 /*****************************************************
201 * ITfInputProcessorProfiles functions
202 *****************************************************/
203 static HRESULT WINAPI InputProcessorProfiles_Register(
204 ITfInputProcessorProfiles *iface, REFCLSID rclsid)
206 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
211 TRACE("(%p) %s\n",This,debugstr_guid(rclsid));
213 StringFromGUID2(rclsid, buf, 39);
214 sprintfW(fullkey,szwTipfmt,szwSystemTIPKey,buf);
216 if (RegCreateKeyExW(HKEY_LOCAL_MACHINE,fullkey, 0, NULL, 0,
217 KEY_READ | KEY_WRITE, NULL, &tipkey, NULL) != ERROR_SUCCESS)
225 static HRESULT WINAPI InputProcessorProfiles_Unregister(
226 ITfInputProcessorProfiles *iface, REFCLSID rclsid)
230 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
232 TRACE("(%p) %s\n",This,debugstr_guid(rclsid));
234 StringFromGUID2(rclsid, buf, 39);
235 sprintfW(fullkey,szwTipfmt,szwSystemTIPKey,buf);
237 RegDeleteTreeW(HKEY_LOCAL_MACHINE, fullkey);
238 RegDeleteTreeW(HKEY_CURRENT_USER, fullkey);
243 static HRESULT WINAPI InputProcessorProfiles_AddLanguageProfile(
244 ITfInputProcessorProfiles *iface, REFCLSID rclsid,
245 LANGID langid, REFGUID guidProfile, const WCHAR *pchDesc,
246 ULONG cchDesc, const WCHAR *pchIconFile, ULONG cchFile,
253 DWORD disposition = 0;
255 static const WCHAR fmt2[] = {'%','s','\\','0','x','%','0','8','x','\\','%','s',0};
256 static const WCHAR desc[] = {'D','e','s','c','r','i','p','t','i','o','n',0};
257 static const WCHAR icnf[] = {'I','c','o','n','F','i','l','e',0};
258 static const WCHAR icni[] = {'I','c','o','n','I','n','d','e','x',0};
260 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
262 TRACE("(%p) %s %x %s %s %s %i\n",This,debugstr_guid(rclsid), langid,
263 debugstr_guid(guidProfile), debugstr_wn(pchDesc,cchDesc),
264 debugstr_wn(pchIconFile,cchFile),uIconIndex);
266 StringFromGUID2(rclsid, buf, 39);
267 sprintfW(fullkey,szwTipfmt,szwSystemTIPKey,buf);
269 if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,fullkey, 0, KEY_READ | KEY_WRITE,
270 &tipkey ) != ERROR_SUCCESS)
273 StringFromGUID2(guidProfile, buf, 39);
274 sprintfW(fullkey,fmt2,szwLngp,langid,buf);
276 res = RegCreateKeyExW(tipkey,fullkey, 0, NULL, 0, KEY_READ | KEY_WRITE,
277 NULL, &fmtkey, &disposition);
282 RegSetValueExW(fmtkey, desc, 0, REG_SZ, (LPBYTE)pchDesc, cchDesc * sizeof(WCHAR));
283 RegSetValueExW(fmtkey, icnf, 0, REG_SZ, (LPBYTE)pchIconFile, cchFile * sizeof(WCHAR));
284 RegSetValueExW(fmtkey, icni, 0, REG_DWORD, (LPBYTE)&uIconIndex, sizeof(DWORD));
285 if (disposition == REG_CREATED_NEW_KEY)
286 RegSetValueExW(fmtkey, szwEnable, 0, REG_DWORD, (LPBYTE)&zero, sizeof(DWORD));
289 add_userkey(rclsid, langid, guidProfile);
299 static HRESULT WINAPI InputProcessorProfiles_RemoveLanguageProfile(
300 ITfInputProcessorProfiles *iface, REFCLSID rclsid, LANGID langid,
303 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
304 FIXME("STUB:(%p)\n",This);
308 static HRESULT WINAPI InputProcessorProfiles_EnumInputProcessorInfo(
309 ITfInputProcessorProfiles *iface, IEnumGUID **ppEnum)
311 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
312 TRACE("(%p) %p\n",This,ppEnum);
313 return ProfilesEnumGuid_Constructor(ppEnum);
316 static HRESULT WINAPI InputProcessorProfiles_GetDefaultLanguageProfile(
317 ITfInputProcessorProfiles *iface, LANGID langid, REFGUID catid,
318 CLSID *pclsid, GUID *pguidProfile)
325 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
327 TRACE("%p) %x %s %p %p\n",This, langid, debugstr_guid(catid),pclsid,pguidProfile);
329 if (!catid || !pclsid || !pguidProfile)
332 StringFromGUID2(catid, buf, 39);
333 sprintfW(fullkey, szwDefaultFmt, szwSystemCTFKey, szwAssemblies, langid, buf);
335 if (RegOpenKeyExW(HKEY_CURRENT_USER, fullkey, 0, KEY_READ | KEY_WRITE,
336 &hkey ) != ERROR_SUCCESS)
340 res = RegQueryValueExW(hkey, szwDefault, 0, NULL, (LPBYTE)buf, &count);
341 if (res != ERROR_SUCCESS)
346 CLSIDFromString(buf,pclsid);
348 res = RegQueryValueExW(hkey, szwProfile, 0, NULL, (LPBYTE)buf, &count);
349 if (res == ERROR_SUCCESS)
350 CLSIDFromString(buf,pguidProfile);
357 static HRESULT WINAPI InputProcessorProfiles_SetDefaultLanguageProfile(
358 ITfInputProcessorProfiles *iface, LANGID langid, REFCLSID rclsid,
359 REFGUID guidProfiles)
361 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
362 FIXME("STUB:(%p)\n",This);
366 static HRESULT WINAPI InputProcessorProfiles_ActivateLanguageProfile(
367 ITfInputProcessorProfiles *iface, REFCLSID rclsid, LANGID langid,
368 REFGUID guidProfiles)
372 TF_LANGUAGEPROFILE LanguageProfile;
373 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
375 TRACE("(%p) %s %x %s\n",This,debugstr_guid(rclsid),langid,debugstr_guid(guidProfiles));
377 if (langid != This->currentLanguage) return E_INVALIDARG;
379 if (get_active_textservice(rclsid,NULL))
381 TRACE("Already Active\n");
385 hr = ITfInputProcessorProfiles_IsEnabledLanguageProfile(iface, rclsid,
386 langid, guidProfiles, &enabled);
387 if (FAILED(hr) || !enabled)
389 TRACE("Not Enabled\n");
393 LanguageProfile.clsid = *rclsid;
394 LanguageProfile.langid = langid;
395 LanguageProfile.guidProfile = *guidProfiles;
397 hr = add_active_textservice(&LanguageProfile);
402 static HRESULT WINAPI InputProcessorProfiles_GetActiveLanguageProfile(
403 ITfInputProcessorProfiles *iface, REFCLSID rclsid, LANGID *plangid,
406 TF_LANGUAGEPROFILE profile;
407 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
409 TRACE("(%p) %s %p %p\n",This,debugstr_guid(rclsid),plangid,pguidProfile);
411 if (!rclsid || !plangid || !pguidProfile)
414 if (get_active_textservice(rclsid, &profile))
416 *plangid = profile.langid;
417 *pguidProfile = profile.guidProfile;
422 *pguidProfile = GUID_NULL;
427 static HRESULT WINAPI InputProcessorProfiles_GetLanguageProfileDescription(
428 ITfInputProcessorProfiles *iface, REFCLSID rclsid, LANGID langid,
429 REFGUID guidProfile, BSTR *pbstrProfile)
431 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
432 FIXME("STUB:(%p)\n",This);
436 static HRESULT WINAPI InputProcessorProfiles_GetCurrentLanguage(
437 ITfInputProcessorProfiles *iface, LANGID *plangid)
439 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
440 TRACE("(%p) 0x%x\n",This,This->currentLanguage);
445 *plangid = This->currentLanguage;
450 static HRESULT WINAPI InputProcessorProfiles_ChangeCurrentLanguage(
451 ITfInputProcessorProfiles *iface, LANGID langid)
453 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
454 FIXME("STUB:(%p)\n",This);
458 static HRESULT WINAPI InputProcessorProfiles_GetLanguageList(
459 ITfInputProcessorProfiles *iface, LANGID **ppLangId, ULONG *pulCount)
461 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
462 FIXME("Semi-STUB:(%p)\n",This);
463 *ppLangId = CoTaskMemAlloc(sizeof(LANGID));
464 **ppLangId = This->currentLanguage;
469 static HRESULT WINAPI InputProcessorProfiles_EnumLanguageProfiles(
470 ITfInputProcessorProfiles *iface, LANGID langid,
471 IEnumTfLanguageProfiles **ppEnum)
473 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
474 TRACE("(%p) %x %p\n",This,langid,ppEnum);
475 return EnumTfLanguageProfiles_Constructor(langid, ppEnum);
478 static HRESULT WINAPI InputProcessorProfiles_EnableLanguageProfile(
479 ITfInputProcessorProfiles *iface, REFCLSID rclsid, LANGID langid,
480 REFGUID guidProfile, BOOL fEnable)
488 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
489 TRACE("(%p) %s %x %s %i\n",This, debugstr_guid(rclsid), langid, debugstr_guid(guidProfile), fEnable);
491 StringFromGUID2(rclsid, buf, 39);
492 StringFromGUID2(guidProfile, buf2, 39);
493 sprintfW(fullkey,szwFullLangfmt,szwSystemTIPKey,buf,szwLngp,langid,buf2);
495 res = RegOpenKeyExW(HKEY_CURRENT_USER, fullkey, 0, KEY_READ | KEY_WRITE, &key);
499 RegSetValueExW(key, szwEnable, 0, REG_DWORD, (LPBYTE)&fEnable, sizeof(DWORD));
508 static HRESULT WINAPI InputProcessorProfiles_IsEnabledLanguageProfile(
509 ITfInputProcessorProfiles *iface, REFCLSID rclsid, LANGID langid,
510 REFGUID guidProfile, BOOL *pfEnable)
518 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
519 TRACE("(%p) %s, %i, %s, %p\n",This,debugstr_guid(rclsid),langid,debugstr_guid(guidProfile),pfEnable);
524 StringFromGUID2(rclsid, buf, 39);
525 StringFromGUID2(guidProfile, buf2, 39);
526 sprintfW(fullkey,szwFullLangfmt,szwSystemTIPKey,buf,szwLngp,langid,buf2);
528 res = RegOpenKeyExW(HKEY_CURRENT_USER, fullkey, 0, KEY_READ | KEY_WRITE, &key);
532 DWORD count = sizeof(DWORD);
533 res = RegQueryValueExW(key, szwEnable, 0, NULL, (LPBYTE)pfEnable, &count);
537 if (res) /* Try Default */
539 res = RegOpenKeyExW(HKEY_LOCAL_MACHINE, fullkey, 0, KEY_READ | KEY_WRITE, &key);
543 DWORD count = sizeof(DWORD);
544 res = RegQueryValueExW(key, szwEnable, 0, NULL, (LPBYTE)pfEnable, &count);
555 static HRESULT WINAPI InputProcessorProfiles_EnableLanguageProfileByDefault(
556 ITfInputProcessorProfiles *iface, REFCLSID rclsid, LANGID langid,
557 REFGUID guidProfile, BOOL fEnable)
565 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
566 TRACE("(%p) %s %x %s %i\n",This,debugstr_guid(rclsid),langid,debugstr_guid(guidProfile),fEnable);
568 StringFromGUID2(rclsid, buf, 39);
569 StringFromGUID2(guidProfile, buf2, 39);
570 sprintfW(fullkey,szwFullLangfmt,szwSystemTIPKey,buf,szwLngp,langid,buf2);
572 res = RegOpenKeyExW(HKEY_LOCAL_MACHINE, fullkey, 0, KEY_READ | KEY_WRITE, &key);
576 RegSetValueExW(key, szwEnable, 0, REG_DWORD, (LPBYTE)&fEnable, sizeof(DWORD));
585 static HRESULT WINAPI InputProcessorProfiles_SubstituteKeyboardLayout(
586 ITfInputProcessorProfiles *iface, REFCLSID rclsid, LANGID langid,
587 REFGUID guidProfile, HKL hKL)
589 InputProcessorProfiles *This = (InputProcessorProfiles*)iface;
590 FIXME("STUB:(%p)\n",This);
595 static const ITfInputProcessorProfilesVtbl InputProcessorProfiles_InputProcessorProfilesVtbl =
597 InputProcessorProfiles_QueryInterface,
598 InputProcessorProfiles_AddRef,
599 InputProcessorProfiles_Release,
601 InputProcessorProfiles_Register,
602 InputProcessorProfiles_Unregister,
603 InputProcessorProfiles_AddLanguageProfile,
604 InputProcessorProfiles_RemoveLanguageProfile,
605 InputProcessorProfiles_EnumInputProcessorInfo,
606 InputProcessorProfiles_GetDefaultLanguageProfile,
607 InputProcessorProfiles_SetDefaultLanguageProfile,
608 InputProcessorProfiles_ActivateLanguageProfile,
609 InputProcessorProfiles_GetActiveLanguageProfile,
610 InputProcessorProfiles_GetLanguageProfileDescription,
611 InputProcessorProfiles_GetCurrentLanguage,
612 InputProcessorProfiles_ChangeCurrentLanguage,
613 InputProcessorProfiles_GetLanguageList,
614 InputProcessorProfiles_EnumLanguageProfiles,
615 InputProcessorProfiles_EnableLanguageProfile,
616 InputProcessorProfiles_IsEnabledLanguageProfile,
617 InputProcessorProfiles_EnableLanguageProfileByDefault,
618 InputProcessorProfiles_SubstituteKeyboardLayout
621 /*****************************************************
622 * ITfSource functions
623 *****************************************************/
624 static HRESULT WINAPI IPPSource_QueryInterface(ITfSource *iface, REFIID iid, LPVOID *ppvOut)
626 InputProcessorProfiles *This = impl_from_ITfSourceVtbl(iface);
627 return InputProcessorProfiles_QueryInterface((ITfInputProcessorProfiles *)This, iid, *ppvOut);
630 static ULONG WINAPI IPPSource_AddRef(ITfSource *iface)
632 InputProcessorProfiles *This = impl_from_ITfSourceVtbl(iface);
633 return InputProcessorProfiles_AddRef((ITfInputProcessorProfiles*)This);
636 static ULONG WINAPI IPPSource_Release(ITfSource *iface)
638 InputProcessorProfiles *This = impl_from_ITfSourceVtbl(iface);
639 return InputProcessorProfiles_Release((ITfInputProcessorProfiles *)This);
642 static WINAPI HRESULT IPPSource_AdviseSink(ITfSource *iface,
643 REFIID riid, IUnknown *punk, DWORD *pdwCookie)
645 InputProcessorProfilesSink *ipps;
646 InputProcessorProfiles *This = impl_from_ITfSourceVtbl(iface);
648 TRACE("(%p) %s %p %p\n",This,debugstr_guid(riid),punk,pdwCookie);
650 if (!riid || !punk || !pdwCookie)
653 if (IsEqualIID(riid, &IID_ITfLanguageProfileNotifySink))
655 ipps = HeapAlloc(GetProcessHeap(),0,sizeof(InputProcessorProfilesSink));
657 return E_OUTOFMEMORY;
658 if (FAILED(IUnknown_QueryInterface(punk, riid, (LPVOID *)&ipps->interfaces.pITfLanguageProfileNotifySink)))
660 HeapFree(GetProcessHeap(),0,ipps);
661 return CONNECT_E_CANNOTCONNECT;
663 list_add_head(&This->LanguageProfileNotifySink,&ipps->entry);
664 *pdwCookie = generate_Cookie(COOKIE_MAGIC_IPPSINK, ipps);
668 FIXME("(%p) Unhandled Sink: %s\n",This,debugstr_guid(riid));
672 TRACE("cookie %x\n",*pdwCookie);
677 static WINAPI HRESULT IPPSource_UnadviseSink(ITfSource *iface, DWORD pdwCookie)
679 InputProcessorProfilesSink *sink;
680 InputProcessorProfiles *This = impl_from_ITfSourceVtbl(iface);
682 TRACE("(%p) %x\n",This,pdwCookie);
684 if (get_Cookie_magic(pdwCookie)!=COOKIE_MAGIC_IPPSINK)
687 sink = (InputProcessorProfilesSink*)remove_Cookie(pdwCookie);
689 return CONNECT_E_NOCONNECTION;
691 list_remove(&sink->entry);
697 static const ITfSourceVtbl InputProcessorProfiles_SourceVtbl =
699 IPPSource_QueryInterface,
703 IPPSource_AdviseSink,
704 IPPSource_UnadviseSink,
707 HRESULT InputProcessorProfiles_Constructor(IUnknown *pUnkOuter, IUnknown **ppOut)
709 InputProcessorProfiles *This;
711 return CLASS_E_NOAGGREGATION;
713 This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(InputProcessorProfiles));
715 return E_OUTOFMEMORY;
717 This->InputProcessorProfilesVtbl= &InputProcessorProfiles_InputProcessorProfilesVtbl;
718 This->SourceVtbl = &InputProcessorProfiles_SourceVtbl;
720 This->currentLanguage = GetUserDefaultLCID();
722 list_init(&This->LanguageProfileNotifySink);
724 TRACE("returning %p\n", This);
725 *ppOut = (IUnknown *)This;
729 /**************************************************
730 * IEnumGUID implementaion for ITfInputProcessorProfiles::EnumInputProcessorInfo
731 **************************************************/
732 static void ProfilesEnumGuid_Destructor(ProfilesEnumGuid *This)
734 TRACE("destroying %p\n", This);
735 RegCloseKey(This->key);
736 HeapFree(GetProcessHeap(),0,This);
739 static HRESULT WINAPI ProfilesEnumGuid_QueryInterface(IEnumGUID *iface, REFIID iid, LPVOID *ppvOut)
741 ProfilesEnumGuid *This = (ProfilesEnumGuid *)iface;
744 if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_IEnumGUID))
751 IUnknown_AddRef(iface);
755 WARN("unsupported interface: %s\n", debugstr_guid(iid));
756 return E_NOINTERFACE;
759 static ULONG WINAPI ProfilesEnumGuid_AddRef(IEnumGUID *iface)
761 ProfilesEnumGuid *This = (ProfilesEnumGuid*)iface;
762 return InterlockedIncrement(&This->refCount);
765 static ULONG WINAPI ProfilesEnumGuid_Release(IEnumGUID *iface)
767 ProfilesEnumGuid *This = (ProfilesEnumGuid *)iface;
770 ret = InterlockedDecrement(&This->refCount);
772 ProfilesEnumGuid_Destructor(This);
776 /*****************************************************
777 * IEnumGuid functions
778 *****************************************************/
779 static HRESULT WINAPI ProfilesEnumGuid_Next( LPENUMGUID iface,
780 ULONG celt, GUID *rgelt, ULONG *pceltFetched)
782 ProfilesEnumGuid *This = (ProfilesEnumGuid *)iface;
785 TRACE("(%p)\n",This);
787 if (rgelt == NULL) return E_POINTER;
789 if (This->key) while (fetched < celt)
796 res = RegEnumKeyExW(This->key, This->next_index, catid, &cName,
797 NULL, NULL, NULL, NULL);
798 if (res != ERROR_SUCCESS && res != ERROR_MORE_DATA) break;
799 ++(This->next_index);
801 hr = CLSIDFromString(catid, rgelt);
802 if (FAILED(hr)) continue;
808 if (pceltFetched) *pceltFetched = fetched;
809 return fetched == celt ? S_OK : S_FALSE;
812 static HRESULT WINAPI ProfilesEnumGuid_Skip( LPENUMGUID iface, ULONG celt)
814 ProfilesEnumGuid *This = (ProfilesEnumGuid *)iface;
815 TRACE("(%p)\n",This);
817 This->next_index += celt;
821 static HRESULT WINAPI ProfilesEnumGuid_Reset( LPENUMGUID iface)
823 ProfilesEnumGuid *This = (ProfilesEnumGuid *)iface;
824 TRACE("(%p)\n",This);
825 This->next_index = 0;
829 static HRESULT WINAPI ProfilesEnumGuid_Clone( LPENUMGUID iface,
832 ProfilesEnumGuid *This = (ProfilesEnumGuid *)iface;
835 TRACE("(%p)\n",This);
837 if (ppenum == NULL) return E_POINTER;
839 res = ProfilesEnumGuid_Constructor(ppenum);
842 ProfilesEnumGuid *new_This = (ProfilesEnumGuid *)*ppenum;
843 new_This->next_index = This->next_index;
848 static const IEnumGUIDVtbl IEnumGUID_Vtbl ={
849 ProfilesEnumGuid_QueryInterface,
850 ProfilesEnumGuid_AddRef,
851 ProfilesEnumGuid_Release,
853 ProfilesEnumGuid_Next,
854 ProfilesEnumGuid_Skip,
855 ProfilesEnumGuid_Reset,
856 ProfilesEnumGuid_Clone
859 static HRESULT ProfilesEnumGuid_Constructor(IEnumGUID **ppOut)
861 ProfilesEnumGuid *This;
863 This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(ProfilesEnumGuid));
865 return E_OUTOFMEMORY;
867 This->Vtbl= &IEnumGUID_Vtbl;
870 if (RegCreateKeyExW(HKEY_LOCAL_MACHINE, szwSystemTIPKey, 0, NULL, 0,
871 KEY_READ | KEY_WRITE, NULL, &This->key, NULL) != ERROR_SUCCESS)
874 TRACE("returning %p\n", This);
875 *ppOut = (IEnumGUID*)This;
879 /**************************************************
880 * IEnumTfLanguageProfiles implementaion
881 **************************************************/
882 static void EnumTfLanguageProfiles_Destructor(EnumTfLanguageProfiles *This)
884 TRACE("destroying %p\n", This);
885 RegCloseKey(This->tipkey);
887 RegCloseKey(This->langkey);
888 ITfCategoryMgr_Release(This->catmgr);
889 HeapFree(GetProcessHeap(),0,This);
892 static HRESULT WINAPI EnumTfLanguageProfiles_QueryInterface(IEnumTfLanguageProfiles *iface, REFIID iid, LPVOID *ppvOut)
894 EnumTfLanguageProfiles *This = (EnumTfLanguageProfiles *)iface;
897 if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_IEnumTfLanguageProfiles))
904 IUnknown_AddRef(iface);
908 WARN("unsupported interface: %s\n", debugstr_guid(iid));
909 return E_NOINTERFACE;
912 static ULONG WINAPI EnumTfLanguageProfiles_AddRef(IEnumTfLanguageProfiles *iface)
914 EnumTfLanguageProfiles *This = (EnumTfLanguageProfiles*)iface;
915 return InterlockedIncrement(&This->refCount);
918 static ULONG WINAPI EnumTfLanguageProfiles_Release(IEnumTfLanguageProfiles *iface)
920 EnumTfLanguageProfiles *This = (EnumTfLanguageProfiles *)iface;
923 ret = InterlockedDecrement(&This->refCount);
925 EnumTfLanguageProfiles_Destructor(This);
929 /*****************************************************
930 * IEnumGuid functions
931 *****************************************************/
932 static INT next_LanguageProfile(EnumTfLanguageProfiles *This, CLSID clsid, TF_LANGUAGEPROFILE *tflp)
940 static const WCHAR fmt[] = {'%','s','\\','%','s','\\','0','x','%','0','8','x',0};
942 if (This->langkey == NULL)
944 sprintfW(fullkey,fmt,This->szwCurrentClsid,szwLngp,This->langid);
945 res = RegOpenKeyExW(This->tipkey, fullkey, 0, KEY_READ | KEY_WRITE, &This->langkey);
948 This->langkey = NULL;
951 This->lang_index = 0;
953 res = RegEnumKeyExW(This->langkey, This->lang_index, profileid, &cName,
954 NULL, NULL, NULL, NULL);
955 if (res != ERROR_SUCCESS && res != ERROR_MORE_DATA)
957 RegCloseKey(This->langkey);
958 This->langkey = NULL;
961 ++(This->lang_index);
965 static const GUID * tipcats[3] = { &GUID_TFCAT_TIP_KEYBOARD,
966 &GUID_TFCAT_TIP_SPEECH,
967 &GUID_TFCAT_TIP_HANDWRITING };
968 res = CLSIDFromString(profileid, &profile);
969 if (FAILED(res)) return 0;
972 tflp->langid = This->langid;
973 tflp->fActive = get_active_textservice(&clsid, NULL);
974 tflp->guidProfile = profile;
975 if (ITfCategoryMgr_FindClosestCategory(This->catmgr, &clsid,
976 &tflp->catid, tipcats, 3) != S_OK)
977 ITfCategoryMgr_FindClosestCategory(This->catmgr, &clsid,
978 &tflp->catid, NULL, 0);
984 static HRESULT WINAPI EnumTfLanguageProfiles_Next(IEnumTfLanguageProfiles *iface,
985 ULONG ulCount, TF_LANGUAGEPROFILE *pProfile, ULONG *pcFetch)
987 EnumTfLanguageProfiles *This = (EnumTfLanguageProfiles *)iface;
990 TRACE("(%p)\n",This);
992 if (pProfile == NULL) return E_POINTER;
994 if (This->tipkey) while (fetched < ulCount)
1001 res = RegEnumKeyExW(This->tipkey, This->tip_index,
1002 This->szwCurrentClsid, &cName, NULL, NULL, NULL, NULL);
1003 if (res != ERROR_SUCCESS && res != ERROR_MORE_DATA) break;
1004 ++(This->tip_index);
1005 hr = CLSIDFromString(This->szwCurrentClsid, &clsid);
1006 if (FAILED(hr)) continue;
1008 while ( fetched < ulCount)
1010 INT res = next_LanguageProfile(This, clsid, pProfile);
1023 if (pcFetch) *pcFetch = fetched;
1024 return fetched == ulCount ? S_OK : S_FALSE;
1027 static HRESULT WINAPI EnumTfLanguageProfiles_Skip( IEnumTfLanguageProfiles* iface, ULONG celt)
1029 EnumTfLanguageProfiles *This = (EnumTfLanguageProfiles *)iface;
1030 FIXME("STUB (%p)\n",This);
1034 static HRESULT WINAPI EnumTfLanguageProfiles_Reset( IEnumTfLanguageProfiles* iface)
1036 EnumTfLanguageProfiles *This = (EnumTfLanguageProfiles *)iface;
1037 TRACE("(%p)\n",This);
1038 This->tip_index = 0;
1040 RegCloseKey(This->langkey);
1041 This->langkey = NULL;
1042 This->lang_index = 0;
1046 static HRESULT WINAPI EnumTfLanguageProfiles_Clone( IEnumTfLanguageProfiles *iface,
1047 IEnumTfLanguageProfiles **ppenum)
1049 EnumTfLanguageProfiles *This = (EnumTfLanguageProfiles *)iface;
1052 TRACE("(%p)\n",This);
1054 if (ppenum == NULL) return E_POINTER;
1056 res = EnumTfLanguageProfiles_Constructor(This->langid, ppenum);
1059 EnumTfLanguageProfiles *new_This = (EnumTfLanguageProfiles *)*ppenum;
1060 new_This->tip_index = This->tip_index;
1061 lstrcpynW(new_This->szwCurrentClsid,This->szwCurrentClsid,39);
1066 static const WCHAR fmt[] = {'%','s','\\','%','s','\\','0','x','%','0','8','x',0};
1068 sprintfW(fullkey,fmt,This->szwCurrentClsid,szwLngp,This->langid);
1069 res = RegOpenKeyExW(new_This->tipkey, fullkey, 0, KEY_READ | KEY_WRITE, &This->langkey);
1070 new_This->lang_index = This->lang_index;
1076 static const IEnumTfLanguageProfilesVtbl IEnumTfLanguageProfiles_Vtbl ={
1077 EnumTfLanguageProfiles_QueryInterface,
1078 EnumTfLanguageProfiles_AddRef,
1079 EnumTfLanguageProfiles_Release,
1081 EnumTfLanguageProfiles_Clone,
1082 EnumTfLanguageProfiles_Next,
1083 EnumTfLanguageProfiles_Reset,
1084 EnumTfLanguageProfiles_Skip
1087 static HRESULT EnumTfLanguageProfiles_Constructor(LANGID langid, IEnumTfLanguageProfiles **ppOut)
1090 EnumTfLanguageProfiles *This;
1092 This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(EnumTfLanguageProfiles));
1094 return E_OUTOFMEMORY;
1096 This->Vtbl= &IEnumTfLanguageProfiles_Vtbl;
1098 This->langid = langid;
1100 hr = CategoryMgr_Constructor(NULL,(IUnknown**)&This->catmgr);
1103 HeapFree(GetProcessHeap(),0,This);
1107 if (RegCreateKeyExW(HKEY_LOCAL_MACHINE, szwSystemTIPKey, 0, NULL, 0,
1108 KEY_READ | KEY_WRITE, NULL, &This->tipkey, NULL) != ERROR_SUCCESS)
1111 TRACE("returning %p\n", This);
1112 *ppOut = (IEnumTfLanguageProfiles*)This;