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 #import "speech_recognition_window_controller.h" | |
6 | |
7 #include "base/logging.h" | |
8 #include "base/strings/sys_string_conversions.h" | |
9 #include "chrome/browser/ui/cocoa/info_bubble_view.h" | |
10 #include "grit/generated_resources.h" | |
11 #include "grit/theme_resources.h" | |
12 #import "skia/ext/skia_utils_mac.h" | |
13 #import "third_party/google_toolbox_for_mac/src/AppKit/GTMUILocalizerAndLayoutTw
eaker.h" | |
14 #include "ui/base/l10n/l10n_util_mac.h" | |
15 #include "ui/base/resource/resource_bundle.h" | |
16 #include "ui/gfx/image/image.h" | |
17 | |
18 const int kBubbleControlVerticalSpacing = 10; // Space between controls. | |
19 const int kBubbleHorizontalMargin = 5; // Space on either sides of controls. | |
20 const int kInstructionLabelMaxWidth = 150; | |
21 | |
22 @interface SpeechRecognitionWindowController (Private) | |
23 - (NSSize)calculateContentSize; | |
24 - (void)layout:(NSSize)size; | |
25 @end | |
26 | |
27 @implementation SpeechRecognitionWindowController | |
28 | |
29 - (id)initWithParentWindow:(NSWindow*)parentWindow | |
30 delegate:(SpeechRecognitionBubbleDelegate*)delegate | |
31 anchoredAt:(NSPoint)anchoredAt { | |
32 if ((self = [super initWithWindowNibPath:@"SpeechRecognitionBubble" | |
33 parentWindow:parentWindow | |
34 anchoredAt:anchoredAt])) { | |
35 DCHECK(delegate); | |
36 delegate_ = delegate; | |
37 displayMode_ = SpeechRecognitionBubbleBase::DISPLAY_MODE_WARM_UP; | |
38 } | |
39 return self; | |
40 } | |
41 | |
42 - (void)awakeFromNib { | |
43 [super awakeFromNib]; | |
44 InfoBubbleView* bubble = [self bubble]; | |
45 [bubble setArrowLocation:info_bubble::kTopLeft]; | |
46 NSSize arrowSize = NSMakeSize(0, info_bubble::kBubbleArrowHeight); | |
47 arrowSize = [bubble convertSize:arrowSize toView:nil]; | |
48 NSPoint anchorPoint = self.anchorPoint; | |
49 anchorPoint.y += arrowSize.height / 2.0; | |
50 self.anchorPoint = anchorPoint; | |
51 } | |
52 | |
53 - (IBAction)cancel:(id)sender { | |
54 delegate_->InfoBubbleButtonClicked(SpeechRecognitionBubble::BUTTON_CANCEL); | |
55 } | |
56 | |
57 - (IBAction)tryAgain:(id)sender { | |
58 delegate_->InfoBubbleButtonClicked(SpeechRecognitionBubble::BUTTON_TRY_AGAIN); | |
59 } | |
60 | |
61 - (IBAction)micSettings:(id)sender { | |
62 [[NSWorkspace sharedWorkspace] openFile: | |
63 @"/System/Library/PreferencePanes/Sound.prefPane"]; | |
64 } | |
65 | |
66 // Calculate the window dimensions to reflect the sum height and max width of | |
67 // all controls, with appropriate spacing between and around them. The returned | |
68 // size is in view coordinates. | |
69 - (NSSize)calculateContentSize { | |
70 [GTMUILocalizerAndLayoutTweaker sizeToFitView:cancelButton_]; | |
71 [GTMUILocalizerAndLayoutTweaker sizeToFitView:tryAgainButton_]; | |
72 [GTMUILocalizerAndLayoutTweaker sizeToFitView:micSettingsButton_]; | |
73 NSSize cancelSize = [cancelButton_ bounds].size; | |
74 NSSize tryAgainSize = [tryAgainButton_ bounds].size; | |
75 CGFloat newHeight = cancelSize.height + kBubbleControlVerticalSpacing; | |
76 CGFloat newWidth = cancelSize.width; | |
77 if (![tryAgainButton_ isHidden]) | |
78 newWidth += tryAgainSize.width; | |
79 | |
80 // The size of the bubble in warm up mode is fixed to be the same as in | |
81 // recording mode, so from warm up it can transition to recording without any | |
82 // UI jank. | |
83 bool isWarmUp = (displayMode_ == | |
84 SpeechRecognitionBubbleBase::DISPLAY_MODE_WARM_UP); | |
85 | |
86 if (![iconImage_ isHidden]) { | |
87 NSSize size = [[iconImage_ image] size]; | |
88 if (isWarmUp) { | |
89 NSImage* volumeIcon = | |
90 ResourceBundle::GetSharedInstance().GetNativeImageNamed( | |
91 IDR_SPEECH_INPUT_MIC_EMPTY).ToNSImage(); | |
92 size = [volumeIcon size]; | |
93 } | |
94 newHeight += size.height; | |
95 newWidth = std::max(newWidth, size.width + 2 * kBubbleHorizontalMargin); | |
96 } | |
97 | |
98 if (![instructionLabel_ isHidden] || isWarmUp) { | |
99 [instructionLabel_ sizeToFit]; | |
100 NSSize textSize = [[instructionLabel_ cell] cellSize]; | |
101 NSRect boundsRect = NSMakeRect(0, 0, kInstructionLabelMaxWidth, | |
102 CGFLOAT_MAX); | |
103 NSSize multiLineSize = | |
104 [[instructionLabel_ cell] cellSizeForBounds:boundsRect]; | |
105 if (textSize.width > multiLineSize.width) | |
106 textSize = multiLineSize; | |
107 newHeight += textSize.height + kBubbleControlVerticalSpacing; | |
108 newWidth = std::max(newWidth, textSize.width); | |
109 } | |
110 | |
111 if (![micSettingsButton_ isHidden]) { | |
112 NSSize size = [micSettingsButton_ bounds].size; | |
113 newHeight += size.height; | |
114 newWidth = std::max(newWidth, size.width); | |
115 } | |
116 | |
117 return NSMakeSize(newWidth + 2 * kBubbleHorizontalMargin, | |
118 newHeight + 3 * kBubbleControlVerticalSpacing); | |
119 } | |
120 | |
121 // Position the controls within the given content area bounds. | |
122 - (void)layout:(NSSize)size { | |
123 int y = kBubbleControlVerticalSpacing; | |
124 | |
125 NSRect cancelRect = [cancelButton_ bounds]; | |
126 | |
127 if ([tryAgainButton_ isHidden]) { | |
128 cancelRect.origin.x = (size.width - NSWidth(cancelRect)) / 2; | |
129 } else { | |
130 NSRect tryAgainRect = [tryAgainButton_ bounds]; | |
131 cancelRect.origin.x = (size.width - NSWidth(cancelRect) - | |
132 NSWidth(tryAgainRect)) / 2; | |
133 tryAgainRect.origin.x = cancelRect.origin.x + NSWidth(cancelRect); | |
134 tryAgainRect.origin.y = y; | |
135 [tryAgainButton_ setFrame:tryAgainRect]; | |
136 } | |
137 cancelRect.origin.y = y; | |
138 | |
139 if (![cancelButton_ isHidden]) { | |
140 [cancelButton_ setFrame:cancelRect]; | |
141 y += NSHeight(cancelRect) + kBubbleControlVerticalSpacing; | |
142 } | |
143 | |
144 NSRect rect; | |
145 if (![micSettingsButton_ isHidden]) { | |
146 rect = [micSettingsButton_ bounds]; | |
147 rect.origin.x = (size.width - NSWidth(rect)) / 2; | |
148 rect.origin.y = y; | |
149 [micSettingsButton_ setFrame:rect]; | |
150 y += rect.size.height + kBubbleControlVerticalSpacing; | |
151 } | |
152 | |
153 if (![instructionLabel_ isHidden]) { | |
154 int spaceForIcon = 0; | |
155 if (![iconImage_ isHidden]) { | |
156 spaceForIcon = [[iconImage_ image] size].height + | |
157 kBubbleControlVerticalSpacing; | |
158 } | |
159 | |
160 rect = NSMakeRect(0, y, size.width, size.height - y - spaceForIcon - | |
161 kBubbleControlVerticalSpacing * 2); | |
162 [instructionLabel_ setFrame:rect]; | |
163 y = size.height - spaceForIcon - kBubbleControlVerticalSpacing; | |
164 } | |
165 | |
166 if (![iconImage_ isHidden]) { | |
167 rect.size = [[iconImage_ image] size]; | |
168 // In warm-up mode only the icon gets displayed so center it vertically. | |
169 if (displayMode_ == SpeechRecognitionBubbleBase::DISPLAY_MODE_WARM_UP) | |
170 y = (size.height - rect.size.height) / 2; | |
171 rect.origin.x = (size.width - NSWidth(rect)) / 2; | |
172 rect.origin.y = y; | |
173 [iconImage_ setFrame:rect]; | |
174 } | |
175 } | |
176 | |
177 - (void)updateLayout:(SpeechRecognitionBubbleBase::DisplayMode)mode | |
178 messageText:(const base::string16&)messageText | |
179 iconImage:(NSImage*)iconImage { | |
180 // The very first time this method is called, the child views would still be | |
181 // uninitialized and null. So we invoke [self window] first and that sets up | |
182 // the child views properly so we can do the layout calculations below. | |
183 NSWindow* window = [self window]; | |
184 displayMode_ = mode; | |
185 BOOL is_message = (mode == SpeechRecognitionBubbleBase::DISPLAY_MODE_MESSAGE); | |
186 BOOL is_recording = | |
187 (mode == SpeechRecognitionBubbleBase::DISPLAY_MODE_RECORDING); | |
188 BOOL is_warm_up = (mode == SpeechRecognitionBubbleBase::DISPLAY_MODE_WARM_UP); | |
189 [iconImage_ setHidden:is_message]; | |
190 [tryAgainButton_ setHidden:!is_message]; | |
191 [micSettingsButton_ setHidden:!is_message]; | |
192 [instructionLabel_ setHidden:!is_message && !is_recording]; | |
193 [cancelButton_ setHidden:is_warm_up]; | |
194 | |
195 // Get the right set of controls to be visible. | |
196 if (is_message) { | |
197 [instructionLabel_ setStringValue:base::SysUTF16ToNSString(messageText)]; | |
198 } else { | |
199 [iconImage_ setImage:iconImage]; | |
200 [instructionLabel_ setStringValue:l10n_util::GetNSString( | |
201 IDS_SPEECH_INPUT_BUBBLE_HEADING)]; | |
202 } | |
203 | |
204 NSSize newSize = [self calculateContentSize]; | |
205 [[self bubble] setFrameSize:newSize]; | |
206 | |
207 NSSize windowDelta = [[window contentView] convertSize:newSize toView:nil]; | |
208 NSRect newFrame = [window frame]; | |
209 newFrame.origin.y -= windowDelta.height - newFrame.size.height; | |
210 newFrame.size = windowDelta; | |
211 [window setFrame:newFrame display:YES]; | |
212 | |
213 [self layout:newSize]; // Layout all the child controls. | |
214 } | |
215 | |
216 - (void)windowWillClose:(NSNotification*)notification { | |
217 delegate_->InfoBubbleFocusChanged(); | |
218 } | |
219 | |
220 - (void)show { | |
221 [self showWindow:nil]; | |
222 } | |
223 | |
224 - (void)hide { | |
225 [[self window] orderOut:nil]; | |
226 } | |
227 | |
228 - (void)setImage:(NSImage*)image { | |
229 [iconImage_ setImage:image]; | |
230 } | |
231 | |
232 @end // implementation SpeechRecognitionWindowController | |
OLD | NEW |