/* WinRT Windows.Media.Speech implementation * * Copyright 2021 RĂ©mi Bernon for CodeWeavers * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "private.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(speech); /* * * IVectorView_VoiceInformation * */ struct voice_information_vector { IVectorView_VoiceInformation IVectorView_VoiceInformation_iface; LONG ref; }; static inline struct voice_information_vector *impl_from_IVectorView_VoiceInformation( IVectorView_VoiceInformation *iface ) { return CONTAINING_RECORD(iface, struct voice_information_vector, IVectorView_VoiceInformation_iface); } static HRESULT WINAPI vector_view_voice_information_QueryInterface( IVectorView_VoiceInformation *iface, REFIID iid, void **out ) { struct voice_information_vector *impl = impl_from_IVectorView_VoiceInformation(iface); TRACE("iface %p, iid %s, out %p stub!\n", iface, debugstr_guid(iid), out); if (IsEqualGUID(iid, &IID_IUnknown) || IsEqualGUID(iid, &IID_IInspectable) || IsEqualGUID(iid, &IID_IVectorView_VoiceInformation)) { IInspectable_AddRef((*out = &impl->IVectorView_VoiceInformation_iface)); return S_OK; } WARN("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid)); *out = NULL; return E_NOINTERFACE; } static ULONG WINAPI vector_view_voice_information_AddRef( IVectorView_VoiceInformation *iface ) { struct voice_information_vector *impl = impl_from_IVectorView_VoiceInformation(iface); ULONG ref = InterlockedIncrement(&impl->ref); TRACE("iface %p, ref %lu.\n", iface, ref); return ref; } static ULONG WINAPI vector_view_voice_information_Release( IVectorView_VoiceInformation *iface ) { struct voice_information_vector *impl = impl_from_IVectorView_VoiceInformation(iface); ULONG ref = InterlockedDecrement(&impl->ref); TRACE("iface %p, ref %lu.\n", iface, ref); return ref; } static HRESULT WINAPI vector_view_voice_information_GetIids( IVectorView_VoiceInformation *iface, ULONG *iid_count, IID **iids ) { FIXME("iface %p, iid_count %p, iids %p stub!\n", iface, iid_count, iids); return E_NOTIMPL; } static HRESULT WINAPI vector_view_voice_information_GetRuntimeClassName( IVectorView_VoiceInformation *iface, HSTRING *class_name ) { FIXME("iface %p, class_name %p stub!\n", iface, class_name); return E_NOTIMPL; } static HRESULT WINAPI vector_view_voice_information_GetTrustLevel( IVectorView_VoiceInformation *iface, TrustLevel *trust_level ) { FIXME("iface %p, trust_level %p stub!\n", iface, trust_level); return E_NOTIMPL; } static HRESULT WINAPI vector_view_voice_information_GetAt( IVectorView_VoiceInformation *iface, UINT32 index, IVoiceInformation **value ) { FIXME("iface %p, index %#x, value %p stub!\n", iface, index, value); *value = NULL; return E_BOUNDS; } static HRESULT WINAPI vector_view_voice_information_get_Size( IVectorView_VoiceInformation *iface, UINT32 *value ) { FIXME("iface %p, value %p stub!\n", iface, value); *value = 0; return S_OK; } static HRESULT WINAPI vector_view_voice_information_IndexOf( IVectorView_VoiceInformation *iface, IVoiceInformation *element, UINT32 *index, BOOLEAN *found ) { FIXME("iface %p, element %p, index %p, found %p stub!\n", iface, element, index, found); *index = 0; *found = FALSE; return S_OK; } static HRESULT WINAPI vector_view_voice_information_GetMany( IVectorView_VoiceInformation *iface, UINT32 start_index, UINT32 items_size, IVoiceInformation **items, UINT *value ) { FIXME("iface %p, start_index %#x, items %p, value %p stub!\n", iface, start_index, items, value); *value = 0; return S_OK; } static const struct IVectorView_VoiceInformationVtbl vector_view_voice_information_vtbl = { vector_view_voice_information_QueryInterface, vector_view_voice_information_AddRef, vector_view_voice_information_Release, /* IInspectable methods */ vector_view_voice_information_GetIids, vector_view_voice_information_GetRuntimeClassName, vector_view_voice_information_GetTrustLevel, /* IVectorView methods */ vector_view_voice_information_GetAt, vector_view_voice_information_get_Size, vector_view_voice_information_IndexOf, vector_view_voice_information_GetMany, }; static struct voice_information_vector all_voices = { {&vector_view_voice_information_vtbl}, 0 }; /* * * ISpeechSynthesisStream * */ struct synthesis_stream { ISpeechSynthesisStream ISpeechSynthesisStream_iface; LONG ref; IVector_IMediaMarker *markers; }; static inline struct synthesis_stream *impl_from_ISpeechSynthesisStream( ISpeechSynthesisStream *iface ) { return CONTAINING_RECORD(iface, struct synthesis_stream, ISpeechSynthesisStream_iface); } HRESULT WINAPI synthesis_stream_QueryInterface( ISpeechSynthesisStream *iface, REFIID iid, void **out ) { struct synthesis_stream *impl = impl_from_ISpeechSynthesisStream(iface); TRACE("iface %p, iid %s, out %p stub!\n", iface, debugstr_guid(iid), out); if (IsEqualGUID(iid, &IID_IUnknown) || IsEqualGUID(iid, &IID_IInspectable) || IsEqualGUID(iid, &IID_IAgileObject) || IsEqualGUID(iid, &IID_ISpeechSynthesisStream)) { IInspectable_AddRef((*out = &impl->ISpeechSynthesisStream_iface)); return S_OK; } FIXME("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid)); *out = NULL; return E_NOINTERFACE; } ULONG WINAPI synthesis_stream_AddRef( ISpeechSynthesisStream *iface ) { struct synthesis_stream *impl = impl_from_ISpeechSynthesisStream(iface); ULONG ref = InterlockedIncrement(&impl->ref); TRACE("iface %p, ref %lu.\n", iface, ref); return ref; } ULONG WINAPI synthesis_stream_Release( ISpeechSynthesisStream *iface ) { struct synthesis_stream *impl = impl_from_ISpeechSynthesisStream(iface); ULONG ref = InterlockedDecrement(&impl->ref); TRACE("iface %p, ref %lu.\n", iface, ref); if (!ref) free(impl); return ref; } HRESULT WINAPI synthesis_stream_GetIids( ISpeechSynthesisStream *iface, ULONG *iid_count, IID **iids ) { FIXME("iface %p, iid_count %p, iids %p stub.\n", iface, iid_count, iids); return E_NOTIMPL; } HRESULT WINAPI synthesis_stream_GetRuntimeClassName( ISpeechSynthesisStream *iface, HSTRING *class_name ) { FIXME("iface %p, class_name %p stub.\n", iface, class_name); return E_NOTIMPL; } HRESULT WINAPI synthesis_stream_GetTrustLevel( ISpeechSynthesisStream *iface, TrustLevel *trust_level ) { FIXME("iface %p, trust_level %p stub.\n", iface, trust_level); return E_NOTIMPL; } HRESULT WINAPI synthesis_stream_get_Markers( ISpeechSynthesisStream *iface, IVectorView_IMediaMarker **value ) { struct synthesis_stream *impl = impl_from_ISpeechSynthesisStream(iface); FIXME("iface %p, value %p stub!\n", iface, value); return IVector_IMediaMarker_GetView(impl->markers, value); } static const struct ISpeechSynthesisStreamVtbl synthesis_stream_vtbl = { /* IUnknown methods */ synthesis_stream_QueryInterface, synthesis_stream_AddRef, synthesis_stream_Release, /* IInspectable methods */ synthesis_stream_GetIids, synthesis_stream_GetRuntimeClassName, synthesis_stream_GetTrustLevel, /* ISpeechSynthesisStream methods */ synthesis_stream_get_Markers }; static HRESULT synthesis_stream_create( ISpeechSynthesisStream **out ) { struct synthesis_stream *impl; struct vector_iids markers_iids = { .iterable = &IID_IIterable_IMediaMarker, .iterator = &IID_IIterator_IMediaMarker, .vector = &IID_IVector_IMediaMarker, .view = &IID_IVectorView_IMediaMarker, }; HRESULT hr; TRACE("out %p.\n", out); if (!(impl = calloc(1, sizeof(*impl)))) { *out = NULL; return E_OUTOFMEMORY; } impl->ISpeechSynthesisStream_iface.lpVtbl = &synthesis_stream_vtbl; impl->ref = 1; if (FAILED(hr = vector_inspectable_create(&markers_iids, (IVector_IInspectable**)&impl->markers))) goto error; TRACE("created ISpeechSynthesisStream %p.\n", impl); *out = &impl->ISpeechSynthesisStream_iface; return S_OK; error: free(impl); return hr; } /* * * SpeechSynthesizer runtimeclass * */ struct synthesizer { ISpeechSynthesizer ISpeechSynthesizer_iface; ISpeechSynthesizer2 ISpeechSynthesizer2_iface; IClosable IClosable_iface; LONG ref; }; /* * * ISpeechSynthesizer for SpeechSynthesizer runtimeclass * */ static inline struct synthesizer *impl_from_ISpeechSynthesizer( ISpeechSynthesizer *iface ) { return CONTAINING_RECORD(iface, struct synthesizer, ISpeechSynthesizer_iface); } static HRESULT WINAPI synthesizer_QueryInterface( ISpeechSynthesizer *iface, REFIID iid, void **out ) { struct synthesizer *impl = impl_from_ISpeechSynthesizer(iface); TRACE("iface %p, iid %s, out %p stub!\n", iface, debugstr_guid(iid), out); if (IsEqualGUID(iid, &IID_IUnknown) || IsEqualGUID(iid, &IID_IInspectable) || IsEqualGUID(iid, &IID_ISpeechSynthesizer)) { IInspectable_AddRef((*out = &impl->ISpeechSynthesizer_iface)); return S_OK; } if (IsEqualGUID(iid, &IID_ISpeechSynthesizer2)) { IInspectable_AddRef((*out = &impl->ISpeechSynthesizer2_iface)); return S_OK; } if (IsEqualGUID(iid, &IID_IClosable)) { IInspectable_AddRef((*out = &impl->IClosable_iface)); return S_OK; } FIXME("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid)); *out = NULL; return E_NOINTERFACE; } static ULONG WINAPI synthesizer_AddRef( ISpeechSynthesizer *iface ) { struct synthesizer *impl = impl_from_ISpeechSynthesizer(iface); ULONG ref = InterlockedIncrement(&impl->ref); TRACE("iface %p, ref %lu.\n", iface, ref); return ref; } static ULONG WINAPI synthesizer_Release( ISpeechSynthesizer *iface ) { struct synthesizer *impl = impl_from_ISpeechSynthesizer(iface); ULONG ref = InterlockedDecrement(&impl->ref); TRACE("iface %p, ref %lu.\n", iface, ref); if (!ref) free(impl); return ref; } static HRESULT WINAPI synthesizer_GetIids( ISpeechSynthesizer *iface, ULONG *iid_count, IID **iids ) { FIXME("iface %p, iid_count %p, iids %p stub.\n", iface, iid_count, iids); return E_NOTIMPL; } static HRESULT WINAPI synthesizer_GetRuntimeClassName( ISpeechSynthesizer *iface, HSTRING *class_name ) { FIXME("iface %p, class_name %p stub.\n", iface, class_name); return E_NOTIMPL; } static HRESULT WINAPI synthesizer_GetTrustLevel( ISpeechSynthesizer *iface, TrustLevel *trust_level ) { FIXME("iface %p, trust_level %p stub.\n", iface, trust_level); return E_NOTIMPL; } static HRESULT synthesizer_synthesize_text_to_stream_async( IInspectable *invoker, IInspectable **result ) { return synthesis_stream_create((ISpeechSynthesisStream **)result); } static HRESULT WINAPI synthesizer_SynthesizeTextToStreamAsync( ISpeechSynthesizer *iface, HSTRING text, IAsyncOperation_SpeechSynthesisStream **operation ) { TRACE("iface %p, text %p, operation %p.\n", iface, text, operation); return async_operation_inspectable_create(&IID_IAsyncOperation_SpeechSynthesisStream, NULL, synthesizer_synthesize_text_to_stream_async, (IAsyncOperation_IInspectable **)operation); } static HRESULT synthesizer_synthesize_ssml_to_stream_async( IInspectable *invoker, IInspectable **result ) { return synthesis_stream_create((ISpeechSynthesisStream **)result); } static HRESULT WINAPI synthesizer_SynthesizeSsmlToStreamAsync( ISpeechSynthesizer *iface, HSTRING ssml, IAsyncOperation_SpeechSynthesisStream **operation ) { TRACE("iface %p, ssml %p, operation %p.\n", iface, ssml, operation); return async_operation_inspectable_create(&IID_IAsyncOperation_SpeechSynthesisStream, NULL, synthesizer_synthesize_ssml_to_stream_async, (IAsyncOperation_IInspectable **)operation); } static HRESULT WINAPI synthesizer_put_Voice( ISpeechSynthesizer *iface, IVoiceInformation *value ) { FIXME("iface %p, value %p stub.\n", iface, value); return E_NOTIMPL; } static HRESULT WINAPI synthesizer_get_Voice( ISpeechSynthesizer *iface, IVoiceInformation **value ) { FIXME("iface %p, value %p stub.\n", iface, value); return E_NOTIMPL; } static const struct ISpeechSynthesizerVtbl synthesizer_vtbl = { /* IUnknown methods */ synthesizer_QueryInterface, synthesizer_AddRef, synthesizer_Release, /* IInspectable methods */ synthesizer_GetIids, synthesizer_GetRuntimeClassName, synthesizer_GetTrustLevel, /* ISpeechSynthesizer methods */ synthesizer_SynthesizeTextToStreamAsync, synthesizer_SynthesizeSsmlToStreamAsync, synthesizer_put_Voice, synthesizer_get_Voice, }; /* * * ISpeechSynthesizer2 for SpeechSynthesizer runtimeclass * */ DEFINE_IINSPECTABLE(synthesizer2, ISpeechSynthesizer2, struct synthesizer, ISpeechSynthesizer_iface) static HRESULT WINAPI synthesizer2_get_Options( ISpeechSynthesizer2 *iface, ISpeechSynthesizerOptions **value ) { FIXME("iface %p, value %p stub.\n", iface, value); return E_NOTIMPL; } static const struct ISpeechSynthesizer2Vtbl synthesizer2_vtbl = { /* IUnknown methods */ synthesizer2_QueryInterface, synthesizer2_AddRef, synthesizer2_Release, /* IInspectable methods */ synthesizer2_GetIids, synthesizer2_GetRuntimeClassName, synthesizer2_GetTrustLevel, /* ISpeechSynthesizer2 methods */ synthesizer2_get_Options, }; /* * * IClosable for SpeechSynthesizer runtimeclass * */ DEFINE_IINSPECTABLE(closable, IClosable, struct synthesizer, ISpeechSynthesizer_iface) static HRESULT WINAPI closable_Close( IClosable *iface ) { FIXME("iface %p stub.\n", iface); return E_NOTIMPL; } static const struct IClosableVtbl closable_vtbl = { /* IUnknown methods */ closable_QueryInterface, closable_AddRef, closable_Release, /* IInspectable methods */ closable_GetIids, closable_GetRuntimeClassName, closable_GetTrustLevel, /* IClosable methods */ closable_Close, }; /* * * Static interfaces for SpeechSynthesizer runtimeclass * */ struct synthesizer_statics { IActivationFactory IActivationFactory_iface; IInstalledVoicesStatic IInstalledVoicesStatic_iface; LONG ref; }; /* * * IActivationFactory for SpeechSynthesizer runtimeclass * */ static inline struct synthesizer_statics *impl_from_IActivationFactory( IActivationFactory *iface ) { return CONTAINING_RECORD(iface, struct synthesizer_statics, IActivationFactory_iface); } static HRESULT WINAPI factory_QueryInterface( IActivationFactory *iface, REFIID iid, void **out ) { struct synthesizer_statics *impl = impl_from_IActivationFactory(iface); TRACE("iface %p, iid %s, out %p stub!\n", iface, debugstr_guid(iid), out); if (IsEqualGUID(iid, &IID_IUnknown) || IsEqualGUID(iid, &IID_IInspectable) || IsEqualGUID(iid, &IID_IAgileObject) || IsEqualGUID(iid, &IID_IActivationFactory)) { IInspectable_AddRef((*out = &impl->IActivationFactory_iface)); return S_OK; } if (IsEqualGUID(iid, &IID_IInstalledVoicesStatic)) { IInspectable_AddRef((*out = &impl->IInstalledVoicesStatic_iface)); return S_OK; } FIXME("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid)); *out = NULL; return E_NOINTERFACE; } static ULONG WINAPI factory_AddRef( IActivationFactory *iface ) { struct synthesizer_statics *impl = impl_from_IActivationFactory(iface); ULONG ref = InterlockedIncrement(&impl->ref); TRACE("iface %p, ref %lu.\n", iface, ref); return ref; } static ULONG WINAPI factory_Release( IActivationFactory *iface ) { struct synthesizer_statics *impl = impl_from_IActivationFactory(iface); ULONG ref = InterlockedDecrement(&impl->ref); TRACE("iface %p, ref %lu.\n", iface, ref); return ref; } static HRESULT WINAPI factory_GetIids( IActivationFactory *iface, ULONG *iid_count, IID **iids ) { FIXME("iface %p, iid_count %p, iids %p stub!\n", iface, iid_count, iids); return E_NOTIMPL; } static HRESULT WINAPI factory_GetRuntimeClassName( IActivationFactory *iface, HSTRING *class_name ) { FIXME("iface %p, class_name %p stub!\n", iface, class_name); return E_NOTIMPL; } static HRESULT WINAPI factory_GetTrustLevel( IActivationFactory *iface, TrustLevel *trust_level ) { FIXME("iface %p, trust_level %p stub!\n", iface, trust_level); return E_NOTIMPL; } static HRESULT WINAPI factory_ActivateInstance( IActivationFactory *iface, IInspectable **instance ) { struct synthesizer *impl; TRACE("iface %p, instance %p.\n", iface, instance); if (!(impl = calloc(1, sizeof(*impl)))) { *instance = NULL; return E_OUTOFMEMORY; } impl->ISpeechSynthesizer_iface.lpVtbl = &synthesizer_vtbl; impl->ISpeechSynthesizer2_iface.lpVtbl = &synthesizer2_vtbl; impl->IClosable_iface.lpVtbl = &closable_vtbl; impl->ref = 1; *instance = (IInspectable *)&impl->ISpeechSynthesizer_iface; return S_OK; } static const struct IActivationFactoryVtbl factory_vtbl = { factory_QueryInterface, factory_AddRef, factory_Release, /* IInspectable methods */ factory_GetIids, factory_GetRuntimeClassName, factory_GetTrustLevel, /* IActivationFactory methods */ factory_ActivateInstance, }; /* * * IInstalledVoicesStatic for SpeechSynthesizer runtimeclass * */ DEFINE_IINSPECTABLE(installed_voices_static, IInstalledVoicesStatic, struct synthesizer_statics, IActivationFactory_iface) static HRESULT WINAPI installed_voices_static_get_AllVoices( IInstalledVoicesStatic *iface, IVectorView_VoiceInformation **value ) { TRACE("iface %p, value %p.\n", iface, value); *value = &all_voices.IVectorView_VoiceInformation_iface; IVectorView_VoiceInformation_AddRef(*value); return S_OK; } static HRESULT WINAPI installed_voices_static_get_DefaultVoice( IInstalledVoicesStatic *iface, IVoiceInformation **value ) { FIXME("iface %p, value %p stub!\n", iface, value); return E_NOTIMPL; } static const struct IInstalledVoicesStaticVtbl installed_voices_static_vtbl = { installed_voices_static_QueryInterface, installed_voices_static_AddRef, installed_voices_static_Release, /* IInspectable methods */ installed_voices_static_GetIids, installed_voices_static_GetRuntimeClassName, installed_voices_static_GetTrustLevel, /* IInstalledVoicesStatic methods */ installed_voices_static_get_AllVoices, installed_voices_static_get_DefaultVoice, }; static struct synthesizer_statics synthesizer_statics = { .IActivationFactory_iface = {&factory_vtbl}, .IInstalledVoicesStatic_iface = {&installed_voices_static_vtbl}, .ref = 1 }; IActivationFactory *synthesizer_factory = &synthesizer_statics.IActivationFactory_iface;