| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
					
						
							|  |  |  | // Use of this source code is governed by a BSD-style license that can be | 
					
						
							|  |  |  | // found in the LICENSE file. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <string> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "base/mac/scoped_nsobject.h" | 
					
						
							|  |  |  | #include "base/memory/singleton.h" | 
					
						
							|  |  |  | #include "base/strings/sys_string_conversions.h" | 
					
						
							|  |  |  | #include "base/values.h" | 
					
						
							|  |  |  | #include "chrome/browser/speech/tts_controller.h" | 
					
						
							|  |  |  | #include "chrome/browser/speech/tts_platform.h" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #import <Cocoa/Cocoa.h> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TtsPlatformImplMac; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @interface ChromeTtsDelegate : NSObject <NSSpeechSynthesizerDelegate> { | 
					
						
							|  |  |  |  @private | 
					
						
							|  |  |  |   TtsPlatformImplMac* ttsImplMac_;  // weak. | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (id)initWithPlatformImplMac:(TtsPlatformImplMac*)ttsImplMac; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Subclass of NSSpeechSynthesizer that takes an utterance | 
					
						
							|  |  |  | // string on initialization, retains it and only allows it | 
					
						
							|  |  |  | // to be spoken once. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // We construct a new NSSpeechSynthesizer for each utterance, for | 
					
						
							|  |  |  | // two reasons: | 
					
						
							|  |  |  | // 1. To associate delegate callbacks with a particular utterance, | 
					
						
							|  |  |  | //    without assuming anything undocumented about the protocol. | 
					
						
							|  |  |  | // 2. To work around http://openradar.appspot.com/radar?id=2854403, | 
					
						
							|  |  |  | //    where Nuance voices don't retain the utterance string and | 
					
						
							|  |  |  | //    crash when trying to call willSpeakWord. | 
					
						
							|  |  |  | @interface SingleUseSpeechSynthesizer : NSSpeechSynthesizer { | 
					
						
							|  |  |  |  @private | 
					
						
							|  |  |  |   base::scoped_nsobject<NSString> utterance_; | 
					
						
							|  |  |  |   bool didSpeak_; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (id)initWithUtterance:(NSString*)utterance; | 
					
						
							|  |  |  | - (bool)startSpeakingRetainedUtterance; | 
					
						
							|  |  |  | - (bool)startSpeakingString:(NSString*)utterance; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TtsPlatformImplMac : public TtsPlatformImpl { | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |   bool PlatformImplAvailable() override { return true; } | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |   bool Speak(int utterance_id, | 
					
						
							|  |  |  |              const std::string& utterance, | 
					
						
							|  |  |  |              const std::string& lang, | 
					
						
							|  |  |  |              const VoiceData& voice, | 
					
						
							|  |  |  |              const UtteranceContinuousParameters& params) override; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-17 16:03:51 -07:00
										 |  |  |   bool StopSpeaking() override; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-17 16:03:51 -07:00
										 |  |  |   void Pause() override; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-17 16:03:51 -07:00
										 |  |  |   void Resume() override; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-17 16:03:51 -07:00
										 |  |  |   bool IsSpeaking() override; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-17 16:03:51 -07:00
										 |  |  |   void GetVoices(std::vector<VoiceData>* out_voices) override; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Called by ChromeTtsDelegate when we get a callback from the | 
					
						
							|  |  |  |   // native speech engine. | 
					
						
							|  |  |  |   void OnSpeechEvent(NSSpeechSynthesizer* sender, | 
					
						
							|  |  |  |                      TtsEventType event_type, | 
					
						
							|  |  |  |                      int char_index, | 
					
						
							|  |  |  |                      const std::string& error_message); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Get the single instance of this class. | 
					
						
							|  |  |  |   static TtsPlatformImplMac* GetInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  private: | 
					
						
							|  |  |  |   TtsPlatformImplMac(); | 
					
						
							| 
									
										
										
										
											2018-04-17 16:03:51 -07:00
										 |  |  |   ~TtsPlatformImplMac() override; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   base::scoped_nsobject<SingleUseSpeechSynthesizer> speech_synthesizer_; | 
					
						
							|  |  |  |   base::scoped_nsobject<ChromeTtsDelegate> delegate_; | 
					
						
							|  |  |  |   int utterance_id_; | 
					
						
							|  |  |  |   std::string utterance_; | 
					
						
							|  |  |  |   int last_char_index_; | 
					
						
							|  |  |  |   bool paused_; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-07 19:56:23 +08:00
										 |  |  |   friend struct base::DefaultSingletonTraits<TtsPlatformImplMac>; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   DISALLOW_COPY_AND_ASSIGN(TtsPlatformImplMac); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // static | 
					
						
							|  |  |  | TtsPlatformImpl* TtsPlatformImpl::GetInstance() { | 
					
						
							|  |  |  |   return TtsPlatformImplMac::GetInstance(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  | bool TtsPlatformImplMac::Speak(int utterance_id, | 
					
						
							|  |  |  |                                const std::string& utterance, | 
					
						
							|  |  |  |                                const std::string& lang, | 
					
						
							|  |  |  |                                const VoiceData& voice, | 
					
						
							|  |  |  |                                const UtteranceContinuousParameters& params) { | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   // TODO: convert SSML to SAPI xml. http://crbug.com/88072 | 
					
						
							|  |  |  |   utterance_ = utterance; | 
					
						
							|  |  |  |   paused_ = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   NSString* utterance_nsstring = | 
					
						
							|  |  |  |       [NSString stringWithUTF8String:utterance_.c_str()]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Deliberately construct a new speech synthesizer every time Speak is | 
					
						
							|  |  |  |   // called, otherwise there's no way to know whether calls to the delegate | 
					
						
							|  |  |  |   // apply to the current utterance or a previous utterance. In | 
					
						
							|  |  |  |   // experimentation, the overhead of constructing and destructing a | 
					
						
							|  |  |  |   // NSSpeechSynthesizer is minimal. | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |   speech_synthesizer_.reset([[SingleUseSpeechSynthesizer alloc] | 
					
						
							|  |  |  |       initWithUtterance:utterance_nsstring]); | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   [speech_synthesizer_ setDelegate:delegate_]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!voice.native_voice_identifier.empty()) { | 
					
						
							|  |  |  |     NSString* native_voice_identifier = | 
					
						
							|  |  |  |         [NSString stringWithUTF8String:voice.native_voice_identifier.c_str()]; | 
					
						
							|  |  |  |     [speech_synthesizer_ setVoice:native_voice_identifier]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   utterance_id_ = utterance_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // TODO: support languages other than the default: crbug.com/88059 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (params.rate >= 0.0) { | 
					
						
							|  |  |  |     // The TTS api defines rate via words per minute. Let 200 be the default. | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |     [speech_synthesizer_ setObject:[NSNumber numberWithInt:params.rate * 200] | 
					
						
							|  |  |  |                        forProperty:NSSpeechRateProperty | 
					
						
							|  |  |  |                              error:nil]; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (params.pitch >= 0.0) { | 
					
						
							|  |  |  |     // The input is a float from 0.0 to 2.0, with 1.0 being the default. | 
					
						
							|  |  |  |     // Get the default pitch for this voice and modulate it by 50% - 150%. | 
					
						
							|  |  |  |     NSError* errorCode; | 
					
						
							|  |  |  |     NSNumber* defaultPitchObj = | 
					
						
							|  |  |  |         [speech_synthesizer_ objectForProperty:NSSpeechPitchBaseProperty | 
					
						
							|  |  |  |                                          error:&errorCode]; | 
					
						
							|  |  |  |     int defaultPitch = defaultPitchObj ? [defaultPitchObj intValue] : 48; | 
					
						
							|  |  |  |     int newPitch = static_cast<int>(defaultPitch * (0.5 * params.pitch + 0.5)); | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |     [speech_synthesizer_ setObject:[NSNumber numberWithInt:newPitch] | 
					
						
							|  |  |  |                        forProperty:NSSpeechPitchBaseProperty | 
					
						
							|  |  |  |                              error:nil]; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (params.volume >= 0.0) { | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |     [speech_synthesizer_ setObject:[NSNumber numberWithFloat:params.volume] | 
					
						
							|  |  |  |                        forProperty:NSSpeechVolumeProperty | 
					
						
							|  |  |  |                              error:nil]; | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool success = [speech_synthesizer_ startSpeakingRetainedUtterance]; | 
					
						
							|  |  |  |   if (success) { | 
					
						
							|  |  |  |     TtsController* controller = TtsController::GetInstance(); | 
					
						
							|  |  |  |     controller->OnTtsEvent(utterance_id_, TTS_EVENT_START, 0, ""); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool TtsPlatformImplMac::StopSpeaking() { | 
					
						
							|  |  |  |   if (speech_synthesizer_.get()) { | 
					
						
							|  |  |  |     [speech_synthesizer_ stopSpeaking]; | 
					
						
							|  |  |  |     speech_synthesizer_.reset(nil); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   paused_ = false; | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TtsPlatformImplMac::Pause() { | 
					
						
							|  |  |  |   if (speech_synthesizer_.get() && utterance_id_ && !paused_) { | 
					
						
							|  |  |  |     [speech_synthesizer_ pauseSpeakingAtBoundary:NSSpeechImmediateBoundary]; | 
					
						
							|  |  |  |     paused_ = true; | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |     TtsController::GetInstance()->OnTtsEvent(utterance_id_, TTS_EVENT_PAUSE, | 
					
						
							|  |  |  |                                              last_char_index_, ""); | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TtsPlatformImplMac::Resume() { | 
					
						
							|  |  |  |   if (speech_synthesizer_.get() && utterance_id_ && paused_) { | 
					
						
							|  |  |  |     [speech_synthesizer_ continueSpeaking]; | 
					
						
							|  |  |  |     paused_ = false; | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |     TtsController::GetInstance()->OnTtsEvent(utterance_id_, TTS_EVENT_RESUME, | 
					
						
							|  |  |  |                                              last_char_index_, ""); | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool TtsPlatformImplMac::IsSpeaking() { | 
					
						
							|  |  |  |   if (speech_synthesizer_) | 
					
						
							|  |  |  |     return [speech_synthesizer_ isSpeaking]; | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TtsPlatformImplMac::GetVoices(std::vector<VoiceData>* outVoices) { | 
					
						
							|  |  |  |   NSArray* voices = [NSSpeechSynthesizer availableVoices]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Create a new temporary array of the available voices with | 
					
						
							|  |  |  |   // the default voice first. | 
					
						
							|  |  |  |   NSMutableArray* orderedVoices = | 
					
						
							|  |  |  |       [NSMutableArray arrayWithCapacity:[voices count]]; | 
					
						
							|  |  |  |   NSString* defaultVoice = [NSSpeechSynthesizer defaultVoice]; | 
					
						
							|  |  |  |   if (defaultVoice) { | 
					
						
							|  |  |  |     [orderedVoices addObject:defaultVoice]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   for (NSString* voiceIdentifier in voices) { | 
					
						
							|  |  |  |     if (![voiceIdentifier isEqualToString:defaultVoice]) | 
					
						
							|  |  |  |       [orderedVoices addObject:voiceIdentifier]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (NSString* voiceIdentifier in orderedVoices) { | 
					
						
							|  |  |  |     outVoices->push_back(VoiceData()); | 
					
						
							|  |  |  |     VoiceData& data = outVoices->back(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NSDictionary* attributes = | 
					
						
							|  |  |  |         [NSSpeechSynthesizer attributesForVoice:voiceIdentifier]; | 
					
						
							|  |  |  |     NSString* name = [attributes objectForKey:NSVoiceName]; | 
					
						
							|  |  |  |     NSString* gender = [attributes objectForKey:NSVoiceGender]; | 
					
						
							|  |  |  |     NSString* localeIdentifier = | 
					
						
							|  |  |  |         [attributes objectForKey:NSVoiceLocaleIdentifier]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     data.native = true; | 
					
						
							|  |  |  |     data.native_voice_identifier = base::SysNSStringToUTF8(voiceIdentifier); | 
					
						
							|  |  |  |     data.name = base::SysNSStringToUTF8(name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NSDictionary* localeComponents = | 
					
						
							|  |  |  |         [NSLocale componentsFromLocaleIdentifier:localeIdentifier]; | 
					
						
							|  |  |  |     NSString* language = [localeComponents objectForKey:NSLocaleLanguageCode]; | 
					
						
							|  |  |  |     NSString* country = [localeComponents objectForKey:NSLocaleCountryCode]; | 
					
						
							|  |  |  |     if (language && country) { | 
					
						
							|  |  |  |       data.lang = | 
					
						
							|  |  |  |           [[NSString stringWithFormat:@"%@-%@", language, country] UTF8String]; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       data.lang = base::SysNSStringToUTF8(language); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if ([gender isEqualToString:NSVoiceGenderMale]) | 
					
						
							|  |  |  |       data.gender = TTS_GENDER_MALE; | 
					
						
							|  |  |  |     else if ([gender isEqualToString:NSVoiceGenderFemale]) | 
					
						
							|  |  |  |       data.gender = TTS_GENDER_FEMALE; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       data.gender = TTS_GENDER_NONE; | 
					
						
							|  |  |  |     data.events.insert(TTS_EVENT_START); | 
					
						
							|  |  |  |     data.events.insert(TTS_EVENT_END); | 
					
						
							|  |  |  |     data.events.insert(TTS_EVENT_WORD); | 
					
						
							|  |  |  |     data.events.insert(TTS_EVENT_ERROR); | 
					
						
							|  |  |  |     data.events.insert(TTS_EVENT_CANCELLED); | 
					
						
							|  |  |  |     data.events.insert(TTS_EVENT_INTERRUPTED); | 
					
						
							|  |  |  |     data.events.insert(TTS_EVENT_PAUSE); | 
					
						
							|  |  |  |     data.events.insert(TTS_EVENT_RESUME); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  | void TtsPlatformImplMac::OnSpeechEvent(NSSpeechSynthesizer* sender, | 
					
						
							|  |  |  |                                        TtsEventType event_type, | 
					
						
							|  |  |  |                                        int char_index, | 
					
						
							|  |  |  |                                        const std::string& error_message) { | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   // Don't send events from an utterance that's already completed. | 
					
						
							|  |  |  |   // This depends on the fact that we construct a new NSSpeechSynthesizer | 
					
						
							|  |  |  |   // each time we call Speak. | 
					
						
							|  |  |  |   if (sender != speech_synthesizer_.get()) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (event_type == TTS_EVENT_END) | 
					
						
							|  |  |  |     char_index = utterance_.size(); | 
					
						
							|  |  |  |   TtsController* controller = TtsController::GetInstance(); | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |   controller->OnTtsEvent(utterance_id_, event_type, char_index, error_message); | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   last_char_index_ = char_index; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TtsPlatformImplMac::TtsPlatformImplMac() { | 
					
						
							|  |  |  |   utterance_id_ = -1; | 
					
						
							|  |  |  |   paused_ = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   delegate_.reset([[ChromeTtsDelegate alloc] initWithPlatformImplMac:this]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  | TtsPlatformImplMac::~TtsPlatformImplMac() {} | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // static | 
					
						
							|  |  |  | TtsPlatformImplMac* TtsPlatformImplMac::GetInstance() { | 
					
						
							| 
									
										
										
										
											2015-12-07 19:56:23 +08:00
										 |  |  |   return base::Singleton<TtsPlatformImplMac>::get(); | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @implementation ChromeTtsDelegate | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (id)initWithPlatformImplMac:(TtsPlatformImplMac*)ttsImplMac { | 
					
						
							|  |  |  |   if ((self = [super init])) { | 
					
						
							|  |  |  |     ttsImplMac_ = ttsImplMac; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return self; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (void)speechSynthesizer:(NSSpeechSynthesizer*)sender | 
					
						
							|  |  |  |         didFinishSpeaking:(BOOL)finished_speaking { | 
					
						
							|  |  |  |   ttsImplMac_->OnSpeechEvent(sender, TTS_EVENT_END, 0, ""); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (void)speechSynthesizer:(NSSpeechSynthesizer*)sender | 
					
						
							|  |  |  |             willSpeakWord:(NSRange)character_range | 
					
						
							|  |  |  |                  ofString:(NSString*)string { | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |   ttsImplMac_->OnSpeechEvent(sender, TTS_EVENT_WORD, character_range.location, | 
					
						
							|  |  |  |                              ""); | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (void)speechSynthesizer:(NSSpeechSynthesizer*)sender | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |     didEncounterErrorAtIndex:(NSUInteger)character_index | 
					
						
							|  |  |  |                     ofString:(NSString*)string | 
					
						
							|  |  |  |                      message:(NSString*)message { | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  |   std::string message_utf8 = base::SysNSStringToUTF8(message); | 
					
						
							|  |  |  |   ttsImplMac_->OnSpeechEvent(sender, TTS_EVENT_ERROR, character_index, | 
					
						
							| 
									
										
										
										
											2018-04-20 14:47:04 -04:00
										 |  |  |                              message_utf8); | 
					
						
							| 
									
										
										
										
											2014-10-07 21:18:44 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @implementation SingleUseSpeechSynthesizer | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (id)initWithUtterance:(NSString*)utterance { | 
					
						
							|  |  |  |   self = [super init]; | 
					
						
							|  |  |  |   if (self) { | 
					
						
							|  |  |  |     utterance_.reset([utterance retain]); | 
					
						
							|  |  |  |     didSpeak_ = false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return self; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (bool)startSpeakingRetainedUtterance { | 
					
						
							|  |  |  |   CHECK(!didSpeak_); | 
					
						
							|  |  |  |   CHECK(utterance_); | 
					
						
							|  |  |  |   didSpeak_ = true; | 
					
						
							|  |  |  |   return [super startSpeakingString:utterance_]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (bool)startSpeakingString:(NSString*)utterance { | 
					
						
							|  |  |  |   CHECK(false); | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @end |