OLD | NEW |
| (Empty) |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #include "chrome/browser/speech/speech_recognition_bubble.h" | |
6 | |
7 #include "base/bind.h" | |
8 #include "base/lazy_instance.h" | |
9 #include "base/message_loop/message_loop.h" | |
10 #include "chrome/browser/tab_contents/tab_util.h" | |
11 #include "content/public/browser/web_contents.h" | |
12 #include "content/public/browser/web_contents_view.h" | |
13 #include "grit/generated_resources.h" | |
14 #include "grit/theme_resources.h" | |
15 #include "ui/base/resource/resource_bundle.h" | |
16 #include "ui/gfx/canvas.h" | |
17 #include "ui/gfx/display.h" | |
18 #include "ui/gfx/image/image_skia_operations.h" | |
19 #include "ui/gfx/rect.h" | |
20 #include "ui/gfx/screen.h" | |
21 | |
22 using content::WebContents; | |
23 | |
24 namespace { | |
25 | |
26 const color_utils::HSL kGrayscaleShift = { -1, 0, 0.6 }; | |
27 const int kWarmingUpAnimationStartMs = 500; | |
28 const int kWarmingUpAnimationStepMs = 100; | |
29 const int kRecognizingAnimationStepMs = 100; | |
30 | |
31 // A lazily initialized singleton to hold all the image used by the speech | |
32 // recognition bubbles and safely destroy them on exit. | |
33 class SpeechRecognitionBubbleImages { | |
34 public: | |
35 const std::vector<gfx::ImageSkia>& spinner() const { return spinner_; } | |
36 const std::vector<gfx::ImageSkia>& warm_up() const { return warm_up_; } | |
37 gfx::ImageSkia* mic_full() const { return mic_full_; } | |
38 gfx::ImageSkia* mic_empty() const { return mic_empty_; } | |
39 gfx::ImageSkia* mic_noise() const { return mic_noise_; } | |
40 gfx::ImageSkia* mic_mask() const { return mic_mask_; } | |
41 | |
42 private: | |
43 // Private constructor to enforce singleton. | |
44 friend struct base::DefaultLazyInstanceTraits<SpeechRecognitionBubbleImages>; | |
45 SpeechRecognitionBubbleImages(); | |
46 | |
47 std::vector<gfx::ImageSkia> spinner_; // Frames for the progress spinner. | |
48 std::vector<gfx::ImageSkia> warm_up_; // Frames for the warm up animation. | |
49 | |
50 // These images are owned by ResourceBundle and need not be destroyed. | |
51 gfx::ImageSkia* mic_full_; // Mic image with full volume. | |
52 gfx::ImageSkia* mic_noise_; // Mic image with full noise volume. | |
53 gfx::ImageSkia* mic_empty_; // Mic image with zero volume. | |
54 gfx::ImageSkia* mic_mask_; // Gradient mask used by the volume indicator. | |
55 }; | |
56 | |
57 SpeechRecognitionBubbleImages::SpeechRecognitionBubbleImages() { | |
58 mic_empty_ = ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed( | |
59 IDR_SPEECH_INPUT_MIC_EMPTY); | |
60 mic_noise_ = ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed( | |
61 IDR_SPEECH_INPUT_MIC_NOISE); | |
62 mic_full_ = ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed( | |
63 IDR_SPEECH_INPUT_MIC_FULL); | |
64 mic_mask_ = ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed( | |
65 IDR_SPEECH_INPUT_MIC_MASK); | |
66 | |
67 // The sprite image consists of all the animation frames put together in one | |
68 // horizontal/wide image. Each animation frame is square in shape within the | |
69 // sprite. | |
70 const gfx::ImageSkia* spinner_image = ui::ResourceBundle::GetSharedInstance(). | |
71 GetImageSkiaNamed(IDR_SPEECH_INPUT_SPINNER); | |
72 int frame_size = spinner_image->height(); | |
73 | |
74 // When recording starts up, it may take a short while (few ms or even a | |
75 // couple of seconds) before the audio device starts really capturing data. | |
76 // This is more apparent on first use. To cover such cases we show a warming | |
77 // up state in the bubble starting with a blank spinner image. If audio data | |
78 // starts coming in within a couple hundred ms, we switch to the recording | |
79 // UI and if it takes longer, we show the real warm up animation frames. | |
80 // This reduces visual jank for the most part. | |
81 SkBitmap empty_spinner; | |
82 empty_spinner.setConfig(SkBitmap::kARGB_8888_Config, frame_size, frame_size); | |
83 empty_spinner.allocPixels(); | |
84 empty_spinner.eraseRGB(255, 255, 255); | |
85 // |empty_spinner| has solid color. Pixel doubling a solid color is ok. | |
86 warm_up_.push_back(gfx::ImageSkia::CreateFrom1xBitmap(empty_spinner)); | |
87 | |
88 for (gfx::Rect src_rect(frame_size, frame_size); | |
89 src_rect.x() < spinner_image->width(); | |
90 src_rect.Offset(frame_size, 0)) { | |
91 gfx::ImageSkia frame = gfx::ImageSkiaOperations::ExtractSubset( | |
92 *spinner_image, src_rect); | |
93 | |
94 // The image created by ExtractSubset just points to the same pixels as | |
95 // the original and adjusts rowBytes accordingly. However that doesn't | |
96 // render properly and gets vertically squished in Linux due to a bug in | |
97 // Skia. Until that gets fixed we work around by taking a real copy of it | |
98 // below as the copied image has the correct rowBytes and renders fine. | |
99 frame.EnsureRepsForSupportedScales(); | |
100 std::vector<gfx::ImageSkiaRep> image_reps = frame.image_reps(); | |
101 gfx::ImageSkia frame_copy; | |
102 for (size_t i = 0; i < image_reps.size(); ++i) { | |
103 const SkBitmap& copy_src = image_reps[i].sk_bitmap(); | |
104 SkBitmap copy_dst; | |
105 copy_src.copyTo(©_dst, kPMColor_SkColorType); | |
106 frame_copy.AddRepresentation(gfx::ImageSkiaRep( | |
107 copy_dst, image_reps[i].scale())); | |
108 } | |
109 spinner_.push_back(frame_copy); | |
110 | |
111 // The warm up spinner animation is a gray scale version of the real one. | |
112 warm_up_.push_back(gfx::ImageSkiaOperations::CreateHSLShiftedImage( | |
113 frame_copy, kGrayscaleShift)); | |
114 } | |
115 } | |
116 | |
117 base::LazyInstance<SpeechRecognitionBubbleImages> g_images = | |
118 LAZY_INSTANCE_INITIALIZER; | |
119 | |
120 } // namespace | |
121 | |
122 SpeechRecognitionBubble::FactoryMethod SpeechRecognitionBubble::factory_ = NULL; | |
123 const int SpeechRecognitionBubble::kBubbleTargetOffsetX = 10; | |
124 | |
125 SpeechRecognitionBubble* SpeechRecognitionBubble::Create( | |
126 int render_process_id, int render_view_id, Delegate* delegate, | |
127 const gfx::Rect& element_rect) { | |
128 WebContents* web_contents = | |
129 tab_util::GetWebContentsByID(render_process_id, render_view_id); | |
130 | |
131 if (factory_) | |
132 return (*factory_)(web_contents, delegate, element_rect); | |
133 | |
134 // Has the tab already closed before bubble create request was processed? | |
135 if (!web_contents) | |
136 return NULL; | |
137 | |
138 return CreateNativeBubble(render_process_id, render_view_id, | |
139 delegate, element_rect); | |
140 } | |
141 | |
142 SpeechRecognitionBubbleBase::SpeechRecognitionBubbleBase( | |
143 int render_process_id, int render_view_id) | |
144 : weak_factory_(this), | |
145 animation_step_(0), | |
146 display_mode_(DISPLAY_MODE_RECORDING), | |
147 render_process_id_(render_process_id), | |
148 render_view_id_(render_view_id), | |
149 scale_(1.0f) { | |
150 WebContents* web_contents = GetWebContents(); | |
151 gfx::NativeView view = | |
152 web_contents ? web_contents->GetView()->GetNativeView() : NULL; | |
153 gfx::Screen* screen = gfx::Screen::GetScreenFor(view); | |
154 gfx::Display display = screen->GetDisplayNearestWindow(view); | |
155 scale_ = display.device_scale_factor(); | |
156 | |
157 const gfx::ImageSkiaRep& rep = | |
158 g_images.Get().mic_empty()->GetRepresentation(scale_); | |
159 mic_image_.reset(new SkBitmap()); | |
160 mic_image_->setConfig(SkBitmap::kARGB_8888_Config, | |
161 rep.pixel_width(), rep.pixel_height()); | |
162 mic_image_->allocPixels(); | |
163 | |
164 buffer_image_.reset(new SkBitmap()); | |
165 buffer_image_->setConfig(SkBitmap::kARGB_8888_Config, | |
166 rep.pixel_width(), rep.pixel_height()); | |
167 buffer_image_->allocPixels(); | |
168 } | |
169 | |
170 SpeechRecognitionBubbleBase::~SpeechRecognitionBubbleBase() { | |
171 // This destructor is added to make sure members such as the scoped_ptr | |
172 // get destroyed here and the derived classes don't have to care about such | |
173 // member variables which they don't use. | |
174 } | |
175 | |
176 void SpeechRecognitionBubbleBase::SetWarmUpMode() { | |
177 weak_factory_.InvalidateWeakPtrs(); | |
178 display_mode_ = DISPLAY_MODE_WARM_UP; | |
179 animation_step_ = 0; | |
180 DoWarmingUpAnimationStep(); | |
181 UpdateLayout(); | |
182 } | |
183 | |
184 void SpeechRecognitionBubbleBase::DoWarmingUpAnimationStep() { | |
185 SetImage(g_images.Get().warm_up()[animation_step_]); | |
186 base::MessageLoop::current()->PostDelayedTask( | |
187 FROM_HERE, | |
188 base::Bind(&SpeechRecognitionBubbleBase::DoWarmingUpAnimationStep, | |
189 weak_factory_.GetWeakPtr()), | |
190 base::TimeDelta::FromMilliseconds( | |
191 animation_step_ == 0 ? kWarmingUpAnimationStartMs | |
192 : kWarmingUpAnimationStepMs)); | |
193 if (++animation_step_ >= static_cast<int>(g_images.Get().warm_up().size())) | |
194 animation_step_ = 1; // Frame 0 is skipped during the animation. | |
195 } | |
196 | |
197 void SpeechRecognitionBubbleBase::SetRecordingMode() { | |
198 weak_factory_.InvalidateWeakPtrs(); | |
199 display_mode_ = DISPLAY_MODE_RECORDING; | |
200 SetInputVolume(0, 0); | |
201 UpdateLayout(); | |
202 } | |
203 | |
204 void SpeechRecognitionBubbleBase::SetRecognizingMode() { | |
205 display_mode_ = DISPLAY_MODE_RECOGNIZING; | |
206 animation_step_ = 0; | |
207 DoRecognizingAnimationStep(); | |
208 UpdateLayout(); | |
209 } | |
210 | |
211 void SpeechRecognitionBubbleBase::DoRecognizingAnimationStep() { | |
212 SetImage(g_images.Get().spinner()[animation_step_]); | |
213 if (++animation_step_ >= static_cast<int>(g_images.Get().spinner().size())) | |
214 animation_step_ = 0; | |
215 base::MessageLoop::current()->PostDelayedTask( | |
216 FROM_HERE, | |
217 base::Bind(&SpeechRecognitionBubbleBase::DoRecognizingAnimationStep, | |
218 weak_factory_.GetWeakPtr()), | |
219 base::TimeDelta::FromMilliseconds(kRecognizingAnimationStepMs)); | |
220 } | |
221 | |
222 void SpeechRecognitionBubbleBase::SetMessage(const base::string16& text) { | |
223 weak_factory_.InvalidateWeakPtrs(); | |
224 message_text_ = text; | |
225 display_mode_ = DISPLAY_MODE_MESSAGE; | |
226 UpdateLayout(); | |
227 } | |
228 | |
229 void SpeechRecognitionBubbleBase::DrawVolumeOverlay(SkCanvas* canvas, | |
230 const gfx::ImageSkia& image, | |
231 float volume) { | |
232 buffer_image_->eraseARGB(0, 0, 0, 0); | |
233 | |
234 int width = mic_image_->width(); | |
235 int height = mic_image_->height(); | |
236 SkCanvas buffer_canvas(*buffer_image_); | |
237 | |
238 buffer_canvas.save(); | |
239 const int kVolumeSteps = 12; | |
240 SkScalar clip_right = | |
241 (((1.0f - volume) * (width * (kVolumeSteps + 1))) - width) / kVolumeSteps; | |
242 buffer_canvas.clipRect(SkRect::MakeLTRB(0, 0, | |
243 SkIntToScalar(width) - clip_right, SkIntToScalar(height))); | |
244 buffer_canvas.drawBitmap(image.GetRepresentation(scale_).sk_bitmap(), 0, 0); | |
245 buffer_canvas.restore(); | |
246 SkPaint multiply_paint; | |
247 multiply_paint.setXfermodeMode(SkXfermode::kModulate_Mode); | |
248 buffer_canvas.drawBitmap( | |
249 g_images.Get().mic_mask()->GetRepresentation(scale_).sk_bitmap(), | |
250 -clip_right, 0, &multiply_paint); | |
251 | |
252 canvas->drawBitmap(*buffer_image_.get(), 0, 0); | |
253 } | |
254 | |
255 void SpeechRecognitionBubbleBase::SetInputVolume(float volume, | |
256 float noise_volume) { | |
257 mic_image_->eraseARGB(0, 0, 0, 0); | |
258 SkCanvas canvas(*mic_image_); | |
259 | |
260 // Draw the empty volume image first and the current volume image on top, | |
261 // and then the noise volume image on top of both. | |
262 canvas.drawBitmap( | |
263 g_images.Get().mic_empty()->GetRepresentation(scale_).sk_bitmap(), | |
264 0, 0); | |
265 DrawVolumeOverlay(&canvas, *g_images.Get().mic_full(), volume); | |
266 DrawVolumeOverlay(&canvas, *g_images.Get().mic_noise(), noise_volume); | |
267 | |
268 gfx::ImageSkia image(gfx::ImageSkiaRep(*mic_image_.get(), scale_)); | |
269 SetImage(image); | |
270 } | |
271 | |
272 WebContents* SpeechRecognitionBubbleBase::GetWebContents() { | |
273 return tab_util::GetWebContentsByID(render_process_id_, render_view_id_); | |
274 } | |
275 | |
276 void SpeechRecognitionBubbleBase::SetImage(const gfx::ImageSkia& image) { | |
277 icon_image_ = image; | |
278 UpdateImage(); | |
279 } | |
280 | |
281 gfx::ImageSkia SpeechRecognitionBubbleBase::icon_image() { | |
282 return icon_image_; | |
283 } | |
OLD | NEW |