1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
|
// Copyright (c) 2009 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 "chrome/browser/cocoa/tab_contents_controller.h"
#import "base/sys_string_conversions.h"
#import "chrome/app/chrome_dll_resource.h"
#import "chrome/browser/command_updater.h"
#import "chrome/browser/location_bar.h"
#import "chrome/browser/tab_contents/tab_contents.h"
#import "chrome/browser/toolbar_model.h"
// For now, tab_contents lives here. TODO(port):fix
#include "chrome/common/temp_scaffolding_stubs.h"
// Names of images in the bundle for the star icon (normal and 'starred').
static NSString* const kStarImageName = @"star";
static NSString* const kStarredImageName = @"starred";
@interface TabContentsController(CommandUpdates)
- (void)enabledStateChangedForCommand:(NSInteger)command enabled:(BOOL)enabled;
@end
@interface TabContentsController(LocationBar)
- (NSString*)locationBarString;
- (void)focusLocationBar;
@end
@interface TabContentsController(Private)
- (void)updateToolbarCommandStatus;
@end
// A C++ bridge class that handles listening for updates to commands and
// passing them back to the controller.
class TabContentsCommandObserver : public CommandUpdater::CommandObserver {
public:
TabContentsCommandObserver(TabContentsController* controller,
CommandUpdater* commands);
~TabContentsCommandObserver();
// Overridden from CommandUpdater::CommandObserver
void EnabledStateChangedForCommand(int command, bool enabled);
private:
TabContentsController* controller_; // weak, owns me
CommandUpdater* commands_; // weak
};
// A C++ bridge class that handles responding to requests from the
// cross-platform code for information about the location bar. Just passes
// everything back to the controller.
class LocationBarBridge : public LocationBar {
public:
LocationBarBridge(TabContentsController* controller);
// Overridden from LocationBar
virtual void ShowFirstRunBubble() { NOTIMPLEMENTED(); }
virtual std::wstring GetInputString() const;
virtual WindowOpenDisposition GetWindowOpenDisposition() const
{ NOTIMPLEMENTED(); return CURRENT_TAB; }
virtual PageTransition::Type GetPageTransition() const
{ NOTIMPLEMENTED(); return 0; }
virtual void AcceptInput() { NOTIMPLEMENTED(); }
virtual void FocusLocation();
virtual void FocusSearch() { NOTIMPLEMENTED(); }
virtual void SaveStateToContents(TabContents* contents) { NOTIMPLEMENTED(); }
private:
TabContentsController* controller_; // weak, owns me
};
@implementation TabContentsController
- (id)initWithNibName:(NSString*)name
bundle:(NSBundle*)bundle
contents:(TabContents*)contents
commands:(CommandUpdater*)commands
toolbarModel:(ToolbarModel*)toolbarModel {
if ((self = [super initWithNibName:name bundle:bundle])) {
commands_ = commands;
if (commands_)
observer_ = new TabContentsCommandObserver(self, commands);
locationBarBridge_ = new LocationBarBridge(self);
contents_ = contents;
toolbarModel_ = toolbarModel;
}
return self;
}
- (void)dealloc {
// make sure our contents have been removed from the window
[[self view] removeFromSuperview];
delete observer_;
delete locationBarBridge_;
[super dealloc];
}
- (void)awakeFromNib {
[contentsBox_ setContentView:contents_->GetNativeView()];
// Provide a starting point since we won't get notifications if the state
// doesn't change between tabs.
[self updateToolbarCommandStatus];
[locationBar_ setStringValue:@"http://dev.chromium.org"];
}
- (LocationBar*)locationBar {
return locationBarBridge_;
}
// Returns YES if the tab represented by this controller is the front-most.
- (BOOL)isCurrentTab {
// We're the current tab if we're in the view hierarchy, otherwise some other
// tab is.
return [[self view] superview] ? YES : NO;
}
// Called when the state for a command changes to |enabled|. Update the
// corresponding UI element.
- (void)enabledStateChangedForCommand:(NSInteger)command enabled:(BOOL)enabled {
// We don't need to update anything if we're not the frontmost tab.
// TODO(pinkerton): i'm worried that observer ordering could cause the
// notification to be sent before we've been put into the view, but we
// appear to be called in the right order so far.
if (![self isCurrentTab])
return;
NSButton* button = nil;
switch (command) {
case IDC_BACK:
button = backButton_;
break;
case IDC_FORWARD:
button = forwardButton_;
break;
case IDC_HOME:
// TODO(pinkerton): add home button
break;
case IDC_STAR:
button = starButton_;
break;
}
[button setEnabled:enabled];
}
- (IBAction)fullScreen:(id)sender {
if ([[self view] isInFullScreenMode]) {
[[self view] exitFullScreenModeWithOptions:nil];
} else {
[[self view] enterFullScreenMode:[NSScreen mainScreen] withOptions:nil];
}
}
// Set the enabled state of the buttons on the toolbar to match the state in
// the controller. We can't only rely on notifications to do this because the
// command model only assumes a single toolbar and won't send notifications if
// the state doesn't change.
- (void)updateToolbarCommandStatus {
[backButton_ setEnabled:commands_->IsCommandEnabled(IDC_BACK) ? YES : NO];
[forwardButton_
setEnabled:commands_->IsCommandEnabled(IDC_FORWARD) ? YES : NO];
[reloadButton_
setEnabled:commands_->IsCommandEnabled(IDC_RELOAD) ? YES : NO];
[starButton_ setEnabled:commands_->IsCommandEnabled(IDC_STAR) ? YES : NO];
}
- (void)willBecomeSelectedTab {
[self updateToolbarCommandStatus];
}
- (void)tabDidChange:(TabContents*)updatedContents {
contents_ = updatedContents;
[contentsBox_ setContentView:contents_->GetNativeView()];
}
- (NSString*)locationBarString {
return [locationBar_ stringValue];
}
- (void)focusLocationBar {
[[locationBar_ window] makeFirstResponder:locationBar_];
}
- (void)updateToolbarWithContents:(TabContents*)tab {
// TODO(pinkerton): there's a lot of ui code in autocomplete_edit.cc
// that we'll want to duplicate. For now, just handle setting the text.
// TODO(pinkerton): update the security lock icon and background color
NSString* urlString = base::SysWideToNSString(toolbarModel_->GetText());
[locationBar_ setStringValue:urlString];
}
- (void)setStarredState:(BOOL)isStarred {
NSString* starImageName = kStarImageName;
if (isStarred)
starImageName = kStarredImageName;
[starButton_ setImage:[NSImage imageNamed:starImageName]];
}
// Return the rect, in WebKit coordinates (flipped), of the window's grow box
// in the coordinate system of the content area of this tab.
- (NSRect)growBoxRect {
NSRect localGrowBox = NSMakeRect(0, 0, 0, 0);
NSView* contentView = contents_->GetNativeView();
if (contentView) {
// For the rect, we start with the grow box view which is a sibling of
// the content view's containing box. It's in the coordinate system of
// the controller view.
localGrowBox = [growBox_ frame];
// The scrollbar assumes that the resizer goes all the way down to the
// bottom corner, so we ignore any y offset to the rect itself and use the
// entire bottom corner.
localGrowBox.origin.y = 0;
// Convert to the content view's coordinates.
localGrowBox = [contentView convertRect:localGrowBox
fromView:[self view]];
// Flip the rect in view coordinates
localGrowBox.origin.y =
[contentView frame].size.height - localGrowBox.origin.y -
localGrowBox.size.height;
}
return localGrowBox;
}
- (void)setIsLoading:(BOOL)isLoading {
NSString* imageName = @"go";
if (isLoading)
imageName = @"stop";
[goButton_ setImage:[NSImage imageNamed:imageName]];
}
@end
//--------------------------------------------------------------------------
TabContentsCommandObserver::TabContentsCommandObserver(
TabContentsController* controller, CommandUpdater* commands)
: controller_(controller), commands_(commands) {
DCHECK(controller_ && commands);
// Register for notifications about state changes for the toolbar buttons
commands_->AddCommandObserver(IDC_BACK, this);
commands_->AddCommandObserver(IDC_FORWARD, this);
commands_->AddCommandObserver(IDC_RELOAD, this);
commands_->AddCommandObserver(IDC_HOME, this);
commands_->AddCommandObserver(IDC_STAR, this);
}
TabContentsCommandObserver::~TabContentsCommandObserver() {
// Unregister the notifications
commands_->RemoveCommandObserver(this);
}
void TabContentsCommandObserver::EnabledStateChangedForCommand(int command,
bool enabled) {
[controller_ enabledStateChangedForCommand:command
enabled:enabled ? YES : NO];
}
//--------------------------------------------------------------------------
LocationBarBridge::LocationBarBridge(TabContentsController* controller)
: controller_(controller) {
}
std::wstring LocationBarBridge::GetInputString() const {
return base::SysNSStringToWide([controller_ locationBarString]);
}
void LocationBarBridge::FocusLocation() {
[controller_ focusLocationBar];
}
|