Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "base/bind.h" | |
| 5 #include "base/memory/singleton.h" | 6 #include "base/memory/singleton.h" |
| 6 #include "base/message_loop.h" | 7 #include "base/message_loop.h" |
| 7 #include "base/string_number_conversions.h" | 8 #include "base/string_number_conversions.h" |
| 8 #include "base/task.h" | 9 #include "base/task.h" |
| 9 #include "chrome/browser/chromeos/cros/cros_library.h" | 10 #include "chrome/browser/chromeos/dbus/dbus_thread_manager.h" |
| 10 #include "chrome/browser/chromeos/cros/speech_synthesis_library.h" | 11 #include "chrome/browser/chromeos/dbus/speech_synthesizer_client.h" |
| 11 #include "chrome/browser/extensions/extension_tts_api_controller.h" | 12 #include "chrome/browser/extensions/extension_tts_api_controller.h" |
| 12 #include "chrome/browser/extensions/extension_tts_api_platform.h" | 13 #include "chrome/browser/extensions/extension_tts_api_platform.h" |
| 13 | 14 |
| 14 using base::DoubleToString; | 15 using base::DoubleToString; |
| 15 | 16 |
| 16 namespace { | 17 namespace { |
| 17 const char kCrosLibraryNotLoadedError[] = "Cros shared library not loaded."; | |
| 18 const int kSpeechCheckDelayIntervalMs = 100; | 18 const int kSpeechCheckDelayIntervalMs = 100; |
| 19 }; | 19 }; |
| 20 | 20 |
| 21 class ExtensionTtsPlatformImplChromeOs : public ExtensionTtsPlatformImpl { | 21 class ExtensionTtsPlatformImplChromeOs : public ExtensionTtsPlatformImpl { |
| 22 public: | 22 public: |
| 23 virtual bool PlatformImplAvailable() { | 23 virtual bool PlatformImplAvailable() { |
| 24 return true; | 24 return true; |
| 25 } | 25 } |
| 26 | 26 |
| 27 virtual bool Speak( | 27 virtual bool Speak( |
| 28 int utterance_id, | 28 int utterance_id, |
| 29 const std::string& utterance, | 29 const std::string& utterance, |
| 30 const std::string& lang, | 30 const std::string& lang, |
| 31 const UtteranceContinuousParameters& params); | 31 const UtteranceContinuousParameters& params); |
| 32 | 32 |
| 33 virtual bool StopSpeaking(); | 33 virtual bool StopSpeaking(); |
| 34 | 34 |
| 35 virtual bool SendsEvent(TtsEventType event_type); | 35 virtual bool SendsEvent(TtsEventType event_type); |
| 36 | 36 |
| 37 // Get the single instance of this class. | 37 // Get the single instance of this class. |
| 38 static ExtensionTtsPlatformImplChromeOs* GetInstance(); | 38 static ExtensionTtsPlatformImplChromeOs* GetInstance(); |
| 39 | 39 |
| 40 private: | 40 private: |
| 41 ExtensionTtsPlatformImplChromeOs() | 41 ExtensionTtsPlatformImplChromeOs() |
| 42 : ALLOW_THIS_IN_INITIALIZER_LIST(method_factory_(this)) {} | 42 : ALLOW_THIS_IN_INITIALIZER_LIST(method_factory_(this)), |
| 43 ALLOW_THIS_IN_INITIALIZER_LIST(weak_ptr_factory_(this)) {} | |
| 43 virtual ~ExtensionTtsPlatformImplChromeOs() {} | 44 virtual ~ExtensionTtsPlatformImplChromeOs() {} |
| 44 | 45 |
| 45 void PollUntilSpeechFinishes(int utterance_id); | 46 void PollUntilSpeechFinishes(int utterance_id); |
| 47 void ContinuePollingIfSpeechIsNotFinished(int utterance_id, bool result); | |
| 46 | 48 |
| 47 void AppendSpeakOption(std::string key, | 49 void AppendSpeakOption(std::string key, |
| 48 std::string value, | 50 std::string value, |
| 49 std::string* options); | 51 std::string* options); |
| 50 | 52 |
| 51 int utterance_id_; | 53 int utterance_id_; |
| 52 int utterance_length_; | 54 int utterance_length_; |
| 53 ScopedRunnableMethodFactory<ExtensionTtsPlatformImplChromeOs> method_factory_; | 55 ScopedRunnableMethodFactory<ExtensionTtsPlatformImplChromeOs> method_factory_; |
| 56 base::WeakPtrFactory<ExtensionTtsPlatformImplChromeOs> weak_ptr_factory_; | |
| 54 | 57 |
| 55 friend struct DefaultSingletonTraits<ExtensionTtsPlatformImplChromeOs>; | 58 friend struct DefaultSingletonTraits<ExtensionTtsPlatformImplChromeOs>; |
| 56 | 59 |
| 57 DISALLOW_COPY_AND_ASSIGN(ExtensionTtsPlatformImplChromeOs); | 60 DISALLOW_COPY_AND_ASSIGN(ExtensionTtsPlatformImplChromeOs); |
| 58 }; | 61 }; |
| 59 | 62 |
| 60 // static | 63 // static |
| 61 ExtensionTtsPlatformImpl* ExtensionTtsPlatformImpl::GetInstance() { | 64 ExtensionTtsPlatformImpl* ExtensionTtsPlatformImpl::GetInstance() { |
| 62 return ExtensionTtsPlatformImplChromeOs::GetInstance(); | 65 return ExtensionTtsPlatformImplChromeOs::GetInstance(); |
| 63 } | 66 } |
| 64 | 67 |
| 65 bool ExtensionTtsPlatformImplChromeOs::Speak( | 68 bool ExtensionTtsPlatformImplChromeOs::Speak( |
| 66 int utterance_id, | 69 int utterance_id, |
| 67 const std::string& utterance, | 70 const std::string& utterance, |
| 68 const std::string& lang, | 71 const std::string& lang, |
| 69 const UtteranceContinuousParameters& params) { | 72 const UtteranceContinuousParameters& params) { |
| 70 chromeos::CrosLibrary* cros_library = chromeos::CrosLibrary::Get(); | |
| 71 if (!cros_library->EnsureLoaded()) { | |
| 72 set_error(kCrosLibraryNotLoadedError); | |
| 73 return false; | |
| 74 } | |
| 75 | |
| 76 utterance_id_ = utterance_id; | 73 utterance_id_ = utterance_id; |
| 77 utterance_length_ = utterance.size(); | 74 utterance_length_ = utterance.size(); |
| 78 | 75 |
| 79 std::string options; | 76 std::string options; |
| 80 | 77 |
| 81 if (!lang.empty()) { | 78 if (!lang.empty()) { |
| 82 AppendSpeakOption( | 79 AppendSpeakOption( |
| 83 chromeos::SpeechSynthesisLibrary::kSpeechPropertyLocale, | 80 chromeos::SpeechSynthesizerClient::kSpeechPropertyLocale, |
| 84 lang, | 81 lang, |
| 85 &options); | 82 &options); |
| 86 } | 83 } |
| 87 | 84 |
| 88 if (params.rate >= 0.0) { | 85 if (params.rate >= 0.0) { |
| 89 AppendSpeakOption( | 86 AppendSpeakOption( |
| 90 chromeos::SpeechSynthesisLibrary::kSpeechPropertyRate, | 87 chromeos::SpeechSynthesizerClient::kSpeechPropertyRate, |
| 91 DoubleToString(params.rate), | 88 DoubleToString(params.rate), |
| 92 &options); | 89 &options); |
| 93 } | 90 } |
| 94 | 91 |
| 95 if (params.pitch >= 0.0) { | 92 if (params.pitch >= 0.0) { |
| 96 // The TTS service allows a range of 0 to 2 for speech pitch. | 93 // The TTS service allows a range of 0 to 2 for speech pitch. |
| 97 AppendSpeakOption( | 94 AppendSpeakOption( |
| 98 chromeos::SpeechSynthesisLibrary::kSpeechPropertyPitch, | 95 chromeos::SpeechSynthesizerClient::kSpeechPropertyPitch, |
| 99 DoubleToString(params.pitch), | 96 DoubleToString(params.pitch), |
| 100 &options); | 97 &options); |
| 101 } | 98 } |
| 102 | 99 |
| 103 if (params.volume >= 0.0) { | 100 if (params.volume >= 0.0) { |
| 104 // The Chrome OS TTS service allows a range of 0 to 5 for speech volume, | 101 // The Chrome OS TTS service allows a range of 0 to 5 for speech volume, |
| 105 // but 5 clips, so map to a range of 0...4. | 102 // but 5 clips, so map to a range of 0...4. |
| 106 AppendSpeakOption( | 103 AppendSpeakOption( |
| 107 chromeos::SpeechSynthesisLibrary::kSpeechPropertyVolume, | 104 chromeos::SpeechSynthesizerClient::kSpeechPropertyVolume, |
| 108 DoubleToString(params.volume * 4), | 105 DoubleToString(params.volume * 4), |
| 109 &options); | 106 &options); |
| 110 } | 107 } |
| 111 | 108 |
| 112 if (!options.empty()) { | 109 chromeos::SpeechSynthesizerClient* speech_synthesizer_client = |
| 113 cros_library->GetSpeechSynthesisLibrary()->SetSpeakProperties( | 110 chromeos::DBusThreadManager::Get()->speech_synthesizer_client(); |
| 114 options.c_str()); | |
| 115 } | |
| 116 | 111 |
| 117 bool result = | 112 if (!options.empty()) |
| 118 cros_library->GetSpeechSynthesisLibrary()->Speak(utterance.c_str()); | 113 speech_synthesizer_client->SetSpeakProperties(options); |
| 119 | 114 |
| 120 if (result) { | 115 speech_synthesizer_client->Speak(utterance); |
|
satorux1
2011/10/18 20:49:25
I was a bit worried about the change here as Speak
hashimoto
2011/10/19 00:08:47
Yes, this change keeps the semantics, nothing is c
| |
| 121 ExtensionTtsController* controller = ExtensionTtsController::GetInstance(); | 116 ExtensionTtsController* controller = ExtensionTtsController::GetInstance(); |
| 122 controller->OnTtsEvent(utterance_id_, TTS_EVENT_START, 0, std::string()); | 117 controller->OnTtsEvent(utterance_id_, TTS_EVENT_START, 0, std::string()); |
| 123 PollUntilSpeechFinishes(utterance_id_); | 118 PollUntilSpeechFinishes(utterance_id_); |
| 124 } | |
| 125 | 119 |
| 126 return result; | 120 return true; |
| 127 } | 121 } |
| 128 | 122 |
| 129 bool ExtensionTtsPlatformImplChromeOs::StopSpeaking() { | 123 bool ExtensionTtsPlatformImplChromeOs::StopSpeaking() { |
| 130 if (chromeos::CrosLibrary::Get()->EnsureLoaded()) { | 124 chromeos::DBusThreadManager::Get()->speech_synthesizer_client()-> |
| 131 return chromeos::CrosLibrary::Get()->GetSpeechSynthesisLibrary()-> | 125 StopSpeaking(); |
| 132 StopSpeaking(); | 126 return true; |
| 133 } | |
| 134 | |
| 135 set_error(kCrosLibraryNotLoadedError); | |
| 136 return false; | |
| 137 } | 127 } |
| 138 | 128 |
| 139 bool ExtensionTtsPlatformImplChromeOs::SendsEvent(TtsEventType event_type) { | 129 bool ExtensionTtsPlatformImplChromeOs::SendsEvent(TtsEventType event_type) { |
| 140 return (event_type == TTS_EVENT_START || | 130 return (event_type == TTS_EVENT_START || |
| 141 event_type == TTS_EVENT_END || | 131 event_type == TTS_EVENT_END || |
| 142 event_type == TTS_EVENT_ERROR); | 132 event_type == TTS_EVENT_ERROR); |
| 143 } | 133 } |
| 144 | 134 |
| 145 void ExtensionTtsPlatformImplChromeOs::PollUntilSpeechFinishes( | 135 void ExtensionTtsPlatformImplChromeOs::PollUntilSpeechFinishes( |
| 146 int utterance_id) { | 136 int utterance_id) { |
| 147 if (utterance_id != utterance_id_) { | 137 if (utterance_id != utterance_id_) { |
| 148 // This utterance must have been interrupted or cancelled. | 138 // This utterance must have been interrupted or cancelled. |
| 149 return; | 139 return; |
| 150 } | 140 } |
| 141 chromeos::SpeechSynthesizerClient* speech_synthesizer_client = | |
| 142 chromeos::DBusThreadManager::Get()->speech_synthesizer_client(); | |
| 143 speech_synthesizer_client->IsSpeaking(base::Bind( | |
| 144 &ExtensionTtsPlatformImplChromeOs::ContinuePollingIfSpeechIsNotFinished, | |
| 145 weak_ptr_factory_.GetWeakPtr(), utterance_id)); | |
|
satorux1
2011/10/18 20:49:25
Nice! It's now asynchronous thus not blocking UI.
hashimoto
2011/10/19 00:08:47
I tried to make IsSpeaking synchronous, but gave u
| |
| 146 } | |
| 151 | 147 |
| 152 chromeos::CrosLibrary* cros_library = chromeos::CrosLibrary::Get(); | 148 void ExtensionTtsPlatformImplChromeOs::ContinuePollingIfSpeechIsNotFinished( |
| 153 ExtensionTtsController* controller = ExtensionTtsController::GetInstance(); | 149 int utterance_id, bool is_speaking) { |
| 154 | 150 if (utterance_id != utterance_id_) { |
| 155 if (!cros_library->EnsureLoaded()) { | 151 // This utterance must have been interrupted or cancelled. |
| 156 controller->OnTtsEvent( | |
| 157 utterance_id_, TTS_EVENT_ERROR, 0, kCrosLibraryNotLoadedError); | |
| 158 return; | 152 return; |
| 159 } | 153 } |
| 160 | 154 if (!is_speaking) { |
| 161 if (!cros_library->GetSpeechSynthesisLibrary()->IsSpeaking()) { | 155 ExtensionTtsController* controller = ExtensionTtsController::GetInstance(); |
| 162 controller->OnTtsEvent( | 156 controller->OnTtsEvent( |
| 163 utterance_id_, TTS_EVENT_END, utterance_length_, std::string()); | 157 utterance_id_, TTS_EVENT_END, utterance_length_, std::string()); |
| 164 return; | 158 return; |
| 165 } | 159 } |
| 166 | 160 // Contniue polling. |
| 167 MessageLoop::current()->PostDelayedTask( | 161 MessageLoop::current()->PostDelayedTask( |
| 168 FROM_HERE, method_factory_.NewRunnableMethod( | 162 FROM_HERE, method_factory_.NewRunnableMethod( |
| 169 &ExtensionTtsPlatformImplChromeOs::PollUntilSpeechFinishes, | 163 &ExtensionTtsPlatformImplChromeOs::PollUntilSpeechFinishes, |
| 170 utterance_id), | 164 utterance_id), |
| 171 kSpeechCheckDelayIntervalMs); | 165 kSpeechCheckDelayIntervalMs); |
| 172 } | 166 } |
| 173 | 167 |
| 174 void ExtensionTtsPlatformImplChromeOs::AppendSpeakOption( | 168 void ExtensionTtsPlatformImplChromeOs::AppendSpeakOption( |
| 175 std::string key, | 169 std::string key, |
| 176 std::string value, | 170 std::string value, |
| 177 std::string* options) { | 171 std::string* options) { |
| 178 *options += | 172 *options += |
| 179 key + | 173 key + |
| 180 chromeos::SpeechSynthesisLibrary::kSpeechPropertyEquals + | 174 chromeos::SpeechSynthesizerClient::kSpeechPropertyEquals + |
| 181 value + | 175 value + |
| 182 chromeos::SpeechSynthesisLibrary::kSpeechPropertyDelimiter; | 176 chromeos::SpeechSynthesizerClient::kSpeechPropertyDelimiter; |
| 183 } | 177 } |
| 184 | 178 |
| 185 // static | 179 // static |
| 186 ExtensionTtsPlatformImplChromeOs* | 180 ExtensionTtsPlatformImplChromeOs* |
| 187 ExtensionTtsPlatformImplChromeOs::GetInstance() { | 181 ExtensionTtsPlatformImplChromeOs::GetInstance() { |
| 188 return Singleton<ExtensionTtsPlatformImplChromeOs>::get(); | 182 return Singleton<ExtensionTtsPlatformImplChromeOs>::get(); |
| 189 } | 183 } |
| OLD | NEW |