352 lines
11 KiB
Text
352 lines
11 KiB
Text
// 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:
|
|
virtual bool PlatformImplAvailable() override {
|
|
return true;
|
|
}
|
|
|
|
virtual bool Speak(
|
|
int utterance_id,
|
|
const std::string& utterance,
|
|
const std::string& lang,
|
|
const VoiceData& voice,
|
|
const UtteranceContinuousParameters& params) override;
|
|
|
|
virtual bool StopSpeaking() override;
|
|
|
|
virtual void Pause() override;
|
|
|
|
virtual void Resume() override;
|
|
|
|
virtual bool IsSpeaking() override;
|
|
|
|
virtual void GetVoices(std::vector<VoiceData>* out_voices) override;
|
|
|
|
// 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();
|
|
virtual ~TtsPlatformImplMac();
|
|
|
|
base::scoped_nsobject<SingleUseSpeechSynthesizer> speech_synthesizer_;
|
|
base::scoped_nsobject<ChromeTtsDelegate> delegate_;
|
|
int utterance_id_;
|
|
std::string utterance_;
|
|
int last_char_index_;
|
|
bool paused_;
|
|
|
|
friend struct DefaultSingletonTraits<TtsPlatformImplMac>;
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(TtsPlatformImplMac);
|
|
};
|
|
|
|
// static
|
|
TtsPlatformImpl* TtsPlatformImpl::GetInstance() {
|
|
return TtsPlatformImplMac::GetInstance();
|
|
}
|
|
|
|
bool TtsPlatformImplMac::Speak(
|
|
int utterance_id,
|
|
const std::string& utterance,
|
|
const std::string& lang,
|
|
const VoiceData& voice,
|
|
const UtteranceContinuousParameters& params) {
|
|
// 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.
|
|
speech_synthesizer_.reset(
|
|
[[SingleUseSpeechSynthesizer alloc]
|
|
initWithUtterance:utterance_nsstring]);
|
|
[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.
|
|
[speech_synthesizer_
|
|
setObject:[NSNumber numberWithInt:params.rate * 200]
|
|
forProperty:NSSpeechRateProperty error:nil];
|
|
}
|
|
|
|
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));
|
|
[speech_synthesizer_
|
|
setObject:[NSNumber numberWithInt:newPitch]
|
|
forProperty:NSSpeechPitchBaseProperty error:nil];
|
|
}
|
|
|
|
if (params.volume >= 0.0) {
|
|
[speech_synthesizer_
|
|
setObject: [NSNumber numberWithFloat:params.volume]
|
|
forProperty:NSSpeechVolumeProperty error:nil];
|
|
}
|
|
|
|
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;
|
|
TtsController::GetInstance()->OnTtsEvent(
|
|
utterance_id_, TTS_EVENT_PAUSE, last_char_index_, "");
|
|
}
|
|
}
|
|
|
|
void TtsPlatformImplMac::Resume() {
|
|
if (speech_synthesizer_.get() && utterance_id_ && paused_) {
|
|
[speech_synthesizer_ continueSpeaking];
|
|
paused_ = false;
|
|
TtsController::GetInstance()->OnTtsEvent(
|
|
utterance_id_, TTS_EVENT_RESUME, last_char_index_, "");
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
}
|
|
|
|
void TtsPlatformImplMac::OnSpeechEvent(
|
|
NSSpeechSynthesizer* sender,
|
|
TtsEventType event_type,
|
|
int char_index,
|
|
const std::string& error_message) {
|
|
// 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();
|
|
controller->OnTtsEvent(
|
|
utterance_id_, event_type, char_index, error_message);
|
|
last_char_index_ = char_index;
|
|
}
|
|
|
|
TtsPlatformImplMac::TtsPlatformImplMac() {
|
|
utterance_id_ = -1;
|
|
paused_ = false;
|
|
|
|
delegate_.reset([[ChromeTtsDelegate alloc] initWithPlatformImplMac:this]);
|
|
}
|
|
|
|
TtsPlatformImplMac::~TtsPlatformImplMac() {
|
|
}
|
|
|
|
// static
|
|
TtsPlatformImplMac* TtsPlatformImplMac::GetInstance() {
|
|
return Singleton<TtsPlatformImplMac>::get();
|
|
}
|
|
|
|
@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 {
|
|
ttsImplMac_->OnSpeechEvent(sender, TTS_EVENT_WORD,
|
|
character_range.location, "");
|
|
}
|
|
|
|
- (void)speechSynthesizer:(NSSpeechSynthesizer*)sender
|
|
didEncounterErrorAtIndex:(NSUInteger)character_index
|
|
ofString:(NSString*)string
|
|
message:(NSString*)message {
|
|
std::string message_utf8 = base::SysNSStringToUTF8(message);
|
|
ttsImplMac_->OnSpeechEvent(sender, TTS_EVENT_ERROR, character_index,
|
|
message_utf8);
|
|
}
|
|
|
|
@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
|