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.
5 #import <Cocoa/Cocoa.h>
7 #include "base/basictypes.h"
8 #include "base/command_line.h"
9 #include "base/mac/scoped_nsobject.h"
10 #include "base/strings/string16.h"
11 #include "base/strings/string_util.h"
12 #include "base/strings/sys_string_conversions.h"
13 #include "base/strings/utf_string_conversions.h"
14 #include "chrome/browser/bookmarks/bookmark_model_factory.h"
15 #include "chrome/browser/extensions/test_extension_system.h"
16 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_bar_constants.h"
17 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_bar_controller.h"
18 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_bar_folder_window.h"
19 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_bar_unittest_helper.h"
20 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_bar_view.h"
21 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_button.h"
22 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_button_cell.h"
23 #include "chrome/browser/ui/cocoa/cocoa_profile_test.h"
24 #import "chrome/browser/ui/cocoa/view_resizer_pong.h"
25 #include "chrome/common/chrome_switches.h"
26 #include "chrome/common/pref_names.h"
27 #include "chrome/test/base/testing_pref_service_syncable.h"
28 #include "chrome/test/base/testing_profile.h"
29 #include "components/bookmarks/browser/bookmark_model.h"
30 #include "components/bookmarks/browser/bookmark_utils.h"
31 #include "components/bookmarks/test/bookmark_test_helpers.h"
32 #include "testing/gtest/include/gtest/gtest.h"
33 #import "testing/gtest_mac.h"
34 #include "testing/platform_test.h"
35 #import "third_party/ocmock/OCMock/OCMock.h"
36 #include "third_party/ocmock/gtest_support.h"
37 #include "ui/base/cocoa/animation_utils.h"
38 #include "ui/base/theme_provider.h"
39 #include "ui/events/test/cocoa_test_event_utils.h"
40 #include "ui/gfx/image/image_skia.h"
42 using base::ASCIIToUTF16;
44 // Unit tests don't need time-consuming asynchronous animations.
45 @interface BookmarkBarControllerTestable : BookmarkBarController {
50 @implementation BookmarkBarControllerTestable
52 - (id)initWithBrowser:(Browser*)browser
53 initialWidth:(CGFloat)initialWidth
54 delegate:(id<BookmarkBarControllerDelegate>)delegate
55 resizeDelegate:(id<ViewResizer>)resizeDelegate {
56 if ((self = [super initWithBrowser:browser
57 initialWidth:initialWidth
59 resizeDelegate:resizeDelegate])) {
60 [self setStateAnimationsEnabled:NO];
61 [self setInnerContentAnimationsEnabled:NO];
68 // Just like a BookmarkBarController but openURL: is stubbed out.
69 @interface BookmarkBarControllerNoOpen : BookmarkBarControllerTestable {
71 std::vector<GURL> urls_;
72 std::vector<WindowOpenDisposition> dispositions_;
76 @implementation BookmarkBarControllerNoOpen
77 - (void)openURL:(GURL)url disposition:(WindowOpenDisposition)disposition {
79 dispositions_.push_back(disposition);
83 dispositions_.clear();
88 // NSCell that is pre-provided with a desired size that becomes the
89 // return value for -(NSSize)cellSize:.
90 @interface CellWithDesiredSize : NSCell {
94 @property (nonatomic, readonly) NSSize cellSize;
97 @implementation CellWithDesiredSize
99 @synthesize cellSize = cellSize_;
101 - (id)initTextCell:(NSString*)string desiredSize:(NSSize)size {
102 if ((self = [super initTextCell:string])) {
110 // Remember the number of times we've gotten a frameDidChange notification.
111 @interface BookmarkBarControllerTogglePong : BookmarkBarControllerNoOpen {
115 @property (nonatomic, readonly) int toggles;
118 @implementation BookmarkBarControllerTogglePong
120 @synthesize toggles = toggles_;
122 - (void)frameDidChange {
128 // Remembers if a notification callback was called.
129 @interface BookmarkBarControllerNotificationPong : BookmarkBarControllerNoOpen {
130 BOOL windowWillCloseReceived_;
131 BOOL windowDidResignKeyReceived_;
133 @property (nonatomic, readonly) BOOL windowWillCloseReceived;
134 @property (nonatomic, readonly) BOOL windowDidResignKeyReceived;
137 @implementation BookmarkBarControllerNotificationPong
138 @synthesize windowWillCloseReceived = windowWillCloseReceived_;
139 @synthesize windowDidResignKeyReceived = windowDidResignKeyReceived_;
141 // Override NSNotificationCenter callback.
142 - (void)parentWindowWillClose:(NSNotification*)notification {
143 windowWillCloseReceived_ = YES;
146 // NSNotificationCenter callback.
147 - (void)parentWindowDidResignKey:(NSNotification*)notification {
148 windowDidResignKeyReceived_ = YES;
152 // Remembers if and what kind of openAll was performed.
153 @interface BookmarkBarControllerOpenAllPong : BookmarkBarControllerNoOpen {
154 WindowOpenDisposition dispositionDetected_;
156 @property (nonatomic) WindowOpenDisposition dispositionDetected;
159 @implementation BookmarkBarControllerOpenAllPong
160 @synthesize dispositionDetected = dispositionDetected_;
162 // Intercede for the openAll:disposition: method.
163 - (void)openAll:(const BookmarkNode*)node
164 disposition:(WindowOpenDisposition)disposition {
165 [self setDispositionDetected:disposition];
170 // Just like a BookmarkBarController but intercedes when providing
171 // pasteboard drag data.
172 @interface BookmarkBarControllerDragData : BookmarkBarControllerTestable {
173 const BookmarkNode* dragDataNode_; // Weak
175 - (void)setDragDataNode:(const BookmarkNode*)node;
178 @implementation BookmarkBarControllerDragData
180 - (id)initWithBrowser:(Browser*)browser
181 initialWidth:(CGFloat)initialWidth
182 delegate:(id<BookmarkBarControllerDelegate>)delegate
183 resizeDelegate:(id<ViewResizer>)resizeDelegate {
184 if ((self = [super initWithBrowser:browser
185 initialWidth:initialWidth
187 resizeDelegate:resizeDelegate])) {
188 dragDataNode_ = NULL;
193 - (void)setDragDataNode:(const BookmarkNode*)node {
194 dragDataNode_ = node;
197 - (std::vector<const BookmarkNode*>)retrieveBookmarkNodeData {
198 std::vector<const BookmarkNode*> dragDataNodes;
200 dragDataNodes.push_back(dragDataNode_);
202 return dragDataNodes;
208 class FakeTheme : public ui::ThemeProvider {
210 FakeTheme(NSColor* color) : color_(color) {}
211 base::scoped_nsobject<NSColor> color_;
213 virtual bool UsingSystemTheme() const OVERRIDE {
216 virtual gfx::ImageSkia* GetImageSkiaNamed(int id) const OVERRIDE {
219 virtual SkColor GetColor(int id) const OVERRIDE { return SkColor(); }
220 virtual int GetDisplayProperty(int id) const OVERRIDE {
223 virtual bool ShouldUseNativeFrame() const OVERRIDE { return false; }
224 virtual bool HasCustomImage(int id) const OVERRIDE { return false; }
225 virtual base::RefCountedMemory* GetRawData(
227 ui::ScaleFactor scale_factor) const OVERRIDE {
230 virtual NSImage* GetNSImageNamed(int id) const OVERRIDE {
233 virtual NSColor* GetNSImageColorNamed(int id) const OVERRIDE {
236 virtual NSColor* GetNSColor(int id) const OVERRIDE {
239 virtual NSColor* GetNSColorTint(int id) const OVERRIDE {
242 virtual NSGradient* GetNSGradient(int id) const OVERRIDE {
248 @interface FakeDragInfo : NSObject {
250 NSPoint dropLocation_;
251 NSDragOperation sourceMask_;
253 @property (nonatomic, assign) NSPoint dropLocation;
254 - (void)setDraggingSourceOperationMask:(NSDragOperation)mask;
257 @implementation FakeDragInfo
259 @synthesize dropLocation = dropLocation_;
262 if ((self = [super init])) {
263 dropLocation_ = NSZeroPoint;
264 sourceMask_ = NSDragOperationMove;
269 // NSDraggingInfo protocol functions.
271 - (id)draggingPasteboard {
275 - (id)draggingSource {
279 - (NSDragOperation)draggingSourceOperationMask {
283 - (NSPoint)draggingLocation {
284 return dropLocation_;
289 - (void)setDraggingSourceOperationMask:(NSDragOperation)mask {
298 class BookmarkBarControllerTestBase : public CocoaProfileTest {
300 base::scoped_nsobject<NSView> parent_view_;
301 base::scoped_nsobject<ViewResizerPong> resizeDelegate_;
303 virtual void SetUp() {
304 CocoaProfileTest::SetUp();
305 ASSERT_TRUE(profile());
307 base::FilePath extension_dir;
308 static_cast<extensions::TestExtensionSystem*>(
309 extensions::ExtensionSystem::Get(profile()))->
310 CreateExtensionService(
311 CommandLine::ForCurrentProcess(),
312 extension_dir, false);
313 resizeDelegate_.reset([[ViewResizerPong alloc] init]);
314 NSRect parent_frame = NSMakeRect(0, 0, 800, 50);
315 parent_view_.reset([[NSView alloc] initWithFrame:parent_frame]);
316 [parent_view_ setHidden:YES];
319 void InstallAndToggleBar(BookmarkBarController* bar) {
320 // Force loading of the nib.
322 // Awkwardness to look like we've been installed.
323 for (NSView* subView in [parent_view_ subviews])
324 [subView removeFromSuperview];
325 [parent_view_ addSubview:[bar view]];
326 NSRect frame = [[[bar view] superview] frame];
327 frame.origin.y = 100;
328 [[[bar view] superview] setFrame:frame];
330 // Make sure it's on in a window so viewDidMoveToWindow is called
331 NSView* contentView = [test_window() contentView];
332 if (![parent_view_ isDescendantOf:contentView])
333 [contentView addSubview:parent_view_];
335 // Make sure it's open so certain things aren't no-ops.
336 [bar updateState:BookmarkBar::SHOW
337 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
341 class BookmarkBarControllerTest : public BookmarkBarControllerTestBase {
343 base::scoped_nsobject<BookmarkBarControllerNoOpen> bar_;
345 virtual void SetUp() OVERRIDE {
346 BookmarkBarControllerTestBase::SetUp();
347 ASSERT_TRUE(browser());
348 AddCommandLineSwitches();
351 [[BookmarkBarControllerNoOpen alloc]
352 initWithBrowser:browser()
353 initialWidth:NSWidth([parent_view_ frame])
355 resizeDelegate:resizeDelegate_.get()]);
357 InstallAndToggleBar(bar_.get());
360 virtual void AddCommandLineSwitches() {}
362 BookmarkBarControllerNoOpen* noOpenBar() {
363 return (BookmarkBarControllerNoOpen*)bar_.get();
367 TEST_F(BookmarkBarControllerTest, ShowWhenShowBookmarkBarTrue) {
368 [bar_ updateState:BookmarkBar::SHOW
369 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
370 EXPECT_TRUE([bar_ isInState:BookmarkBar::SHOW]);
371 EXPECT_FALSE([bar_ isInState:BookmarkBar::DETACHED]);
372 EXPECT_TRUE([bar_ isVisible]);
373 EXPECT_FALSE([bar_ isAnimationRunning]);
374 EXPECT_FALSE([[bar_ view] isHidden]);
375 EXPECT_GT([resizeDelegate_ height], 0);
376 EXPECT_GT([[bar_ view] frame].size.height, 0);
379 TEST_F(BookmarkBarControllerTest, HideWhenShowBookmarkBarFalse) {
380 [bar_ updateState:BookmarkBar::HIDDEN
381 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
382 EXPECT_FALSE([bar_ isInState:BookmarkBar::SHOW]);
383 EXPECT_FALSE([bar_ isInState:BookmarkBar::DETACHED]);
384 EXPECT_FALSE([bar_ isVisible]);
385 EXPECT_FALSE([bar_ isAnimationRunning]);
386 EXPECT_TRUE([[bar_ view] isHidden]);
387 EXPECT_EQ(0, [resizeDelegate_ height]);
388 EXPECT_EQ(0, [[bar_ view] frame].size.height);
391 TEST_F(BookmarkBarControllerTest, HideWhenShowBookmarkBarTrueButDisabled) {
392 [bar_ setBookmarkBarEnabled:NO];
393 [bar_ updateState:BookmarkBar::SHOW
394 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
395 EXPECT_TRUE([bar_ isInState:BookmarkBar::SHOW]);
396 EXPECT_FALSE([bar_ isInState:BookmarkBar::DETACHED]);
397 EXPECT_FALSE([bar_ isVisible]);
398 EXPECT_FALSE([bar_ isAnimationRunning]);
399 EXPECT_TRUE([[bar_ view] isHidden]);
400 EXPECT_EQ(0, [resizeDelegate_ height]);
401 EXPECT_EQ(0, [[bar_ view] frame].size.height);
404 TEST_F(BookmarkBarControllerTest, ShowOnNewTabPage) {
405 [bar_ updateState:BookmarkBar::DETACHED
406 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
407 EXPECT_FALSE([bar_ isInState:BookmarkBar::SHOW]);
408 EXPECT_TRUE([bar_ isInState:BookmarkBar::DETACHED]);
409 EXPECT_TRUE([bar_ isVisible]);
410 EXPECT_FALSE([bar_ isAnimationRunning]);
411 EXPECT_FALSE([[bar_ view] isHidden]);
412 EXPECT_GT([resizeDelegate_ height], 0);
413 EXPECT_GT([[bar_ view] frame].size.height, 0);
415 // Make sure no buttons fall off the bar, either now or when resized
416 // bigger or smaller.
417 CGFloat sizes[] = { 300.0, -100.0, 200.0, -420.0 };
418 CGFloat previousX = 0.0;
419 for (unsigned x = 0; x < arraysize(sizes); x++) {
420 // Confirm the buttons moved from the last check (which may be
421 // init but that's fine).
422 CGFloat newX = [[bar_ offTheSideButton] frame].origin.x;
423 EXPECT_NE(previousX, newX);
426 // Confirm the buttons have a reasonable bounds. Recall that |-frame|
427 // returns rectangles in the superview's coordinates.
428 NSRect buttonViewFrame =
429 [[bar_ buttonView] convertRect:[[bar_ buttonView] frame]
430 fromView:[[bar_ buttonView] superview]];
431 EXPECT_EQ([bar_ buttonView], [[bar_ offTheSideButton] superview]);
432 EXPECT_TRUE(NSContainsRect(buttonViewFrame,
433 [[bar_ offTheSideButton] frame]));
434 EXPECT_EQ([bar_ buttonView], [[bar_ otherBookmarksButton] superview]);
435 EXPECT_TRUE(NSContainsRect(buttonViewFrame,
436 [[bar_ otherBookmarksButton] frame]));
438 // Now move them implicitly.
439 // We confirm FrameChangeNotification works in the next unit test;
440 // we simply assume it works here to resize or reposition the
442 NSRect frame = [[bar_ view] frame];
443 frame.size.width += sizes[x];
444 [[bar_ view] setFrame:frame];
448 // Test whether |-updateState:...| sets currentState as expected. Make
449 // sure things don't crash.
450 TEST_F(BookmarkBarControllerTest, StateChanges) {
451 // First, go in one-at-a-time cycle.
452 [bar_ updateState:BookmarkBar::HIDDEN
453 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
454 EXPECT_EQ(BookmarkBar::HIDDEN, [bar_ currentState]);
455 EXPECT_FALSE([bar_ isVisible]);
456 EXPECT_FALSE([bar_ isAnimationRunning]);
458 [bar_ updateState:BookmarkBar::SHOW
459 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
460 EXPECT_EQ(BookmarkBar::SHOW, [bar_ currentState]);
461 EXPECT_TRUE([bar_ isVisible]);
462 EXPECT_FALSE([bar_ isAnimationRunning]);
464 [bar_ updateState:BookmarkBar::DETACHED
465 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
466 EXPECT_EQ(BookmarkBar::DETACHED, [bar_ currentState]);
467 EXPECT_TRUE([bar_ isVisible]);
468 EXPECT_FALSE([bar_ isAnimationRunning]);
470 // Now try some "jumps".
471 for (int i = 0; i < 2; i++) {
472 [bar_ updateState:BookmarkBar::HIDDEN
473 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
474 EXPECT_EQ(BookmarkBar::HIDDEN, [bar_ currentState]);
475 EXPECT_FALSE([bar_ isVisible]);
476 EXPECT_FALSE([bar_ isAnimationRunning]);
478 [bar_ updateState:BookmarkBar::SHOW
479 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
480 EXPECT_EQ(BookmarkBar::SHOW, [bar_ currentState]);
481 EXPECT_TRUE([bar_ isVisible]);
482 EXPECT_FALSE([bar_ isAnimationRunning]);
485 // Now try some "jumps".
486 for (int i = 0; i < 2; i++) {
487 [bar_ updateState:BookmarkBar::SHOW
488 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
489 EXPECT_EQ(BookmarkBar::SHOW, [bar_ currentState]);
490 EXPECT_TRUE([bar_ isVisible]);
491 EXPECT_FALSE([bar_ isAnimationRunning]);
493 [bar_ updateState:BookmarkBar::DETACHED
494 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
495 EXPECT_EQ(BookmarkBar::DETACHED, [bar_ currentState]);
496 EXPECT_TRUE([bar_ isVisible]);
497 EXPECT_FALSE([bar_ isAnimationRunning]);
501 // Make sure we're watching for frame change notifications.
502 TEST_F(BookmarkBarControllerTest, FrameChangeNotification) {
503 base::scoped_nsobject<BookmarkBarControllerTogglePong> bar;
505 [[BookmarkBarControllerTogglePong alloc]
506 initWithBrowser:browser()
507 initialWidth:100 // arbitrary
509 resizeDelegate:resizeDelegate_.get()]);
510 InstallAndToggleBar(bar.get());
512 // Send a frame did change notification for the pong's view.
513 [[NSNotificationCenter defaultCenter]
514 postNotificationName:NSViewFrameDidChangeNotification
517 EXPECT_GT([bar toggles], 0);
520 // Confirm our "no items" container goes away when we add the 1st
521 // bookmark, and comes back when we delete the bookmark.
522 TEST_F(BookmarkBarControllerTest, NoItemContainerGoesAway) {
523 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
524 const BookmarkNode* bar = model->bookmark_bar_node();
527 BookmarkBarView* view = [bar_ buttonView];
529 NSView* noItemContainer = [view noItemContainer];
530 DCHECK(noItemContainer);
532 EXPECT_FALSE([noItemContainer isHidden]);
533 const BookmarkNode* node = model->AddURL(bar, bar->child_count(),
534 ASCIIToUTF16("title"),
535 GURL("http://www.google.com"));
536 EXPECT_TRUE([noItemContainer isHidden]);
537 model->Remove(bar, bar->GetIndexOf(node));
538 EXPECT_FALSE([noItemContainer isHidden]);
540 // Now try it using a bookmark from the Other Bookmarks.
541 const BookmarkNode* otherBookmarks = model->other_node();
542 node = model->AddURL(otherBookmarks, otherBookmarks->child_count(),
543 ASCIIToUTF16("TheOther"),
544 GURL("http://www.other.com"));
545 EXPECT_FALSE([noItemContainer isHidden]);
546 // Move it from Other Bookmarks to the bar.
547 model->Move(node, bar, 0);
548 EXPECT_TRUE([noItemContainer isHidden]);
549 // Move it back to Other Bookmarks from the bar.
550 model->Move(node, otherBookmarks, 0);
551 EXPECT_FALSE([noItemContainer isHidden]);
554 // Confirm off the side button only enabled when reasonable.
555 TEST_F(BookmarkBarControllerTest, OffTheSideButtonHidden) {
556 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
559 EXPECT_TRUE([bar_ offTheSideButtonIsHidden]);
561 for (int i = 0; i < 2; i++) {
562 bookmarks::AddIfNotBookmarked(
563 model, GURL("http://www.foo.com"), ASCIIToUTF16("small"));
564 EXPECT_TRUE([bar_ offTheSideButtonIsHidden]);
567 const BookmarkNode* parent = model->bookmark_bar_node();
568 for (int i = 0; i < 20; i++) {
569 model->AddURL(parent, parent->child_count(),
570 ASCIIToUTF16("super duper wide title"),
571 GURL("http://superfriends.hall-of-justice.edu"));
573 EXPECT_FALSE([bar_ offTheSideButtonIsHidden]);
575 // Open the "off the side" and start deleting nodes. Make sure
576 // deletion of the last node in "off the side" causes the folder to
578 EXPECT_FALSE([bar_ offTheSideButtonIsHidden]);
579 NSButton* offTheSideButton = [bar_ offTheSideButton];
580 // Open "off the side" menu.
581 [bar_ openOffTheSideFolderFromButton:offTheSideButton];
582 BookmarkBarFolderController* bbfc = [bar_ folderController];
584 [bbfc setIgnoreAnimations:YES];
585 while (!parent->empty()) {
586 // We've completed the job so we're done.
587 if ([bar_ offTheSideButtonIsHidden])
589 // Delete the last button.
590 model->Remove(parent, parent->child_count() - 1);
591 // If last one make sure the menu is closed and the button is hidden.
592 // Else make sure menu stays open.
593 if ([bar_ offTheSideButtonIsHidden]) {
594 EXPECT_FALSE([bar_ folderController]);
596 EXPECT_TRUE([bar_ folderController]);
601 // http://crbug.com/46175 is a crash when deleting bookmarks from the
602 // off-the-side menu while it is open. This test tries to bang hard
603 // in this area to reproduce the crash.
604 TEST_F(BookmarkBarControllerTest, DeleteFromOffTheSideWhileItIsOpen) {
605 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
608 // Add a lot of bookmarks (per the bug).
609 const BookmarkNode* parent = model->bookmark_bar_node();
610 for (int i = 0; i < 100; i++) {
611 std::ostringstream title;
612 title << "super duper wide title " << i;
613 model->AddURL(parent, parent->child_count(), ASCIIToUTF16(title.str()),
614 GURL("http://superfriends.hall-of-justice.edu"));
616 EXPECT_FALSE([bar_ offTheSideButtonIsHidden]);
618 // Open "off the side" menu.
619 NSButton* offTheSideButton = [bar_ offTheSideButton];
620 [bar_ openOffTheSideFolderFromButton:offTheSideButton];
621 BookmarkBarFolderController* bbfc = [bar_ folderController];
623 [bbfc setIgnoreAnimations:YES];
625 // Start deleting items; try and delete randomish ones in case it
626 // makes a difference.
627 int indices[] = { 2, 4, 5, 1, 7, 9, 2, 0, 10, 9 };
628 while (!parent->empty()) {
629 for (unsigned int i = 0; i < arraysize(indices); i++) {
630 if (indices[i] < parent->child_count()) {
631 // First we mouse-enter the button to make things harder.
632 NSArray* buttons = [bbfc buttons];
633 for (BookmarkButton* button in buttons) {
634 if ([button bookmarkNode] == parent->GetChild(indices[i])) {
635 [bbfc mouseEnteredButton:button event:nil];
639 // Then we remove the node. This triggers the button to get
641 model->Remove(parent, indices[i]);
642 // Force visual update which is otherwise delayed.
643 [[bbfc window] displayIfNeeded];
649 // Test whether |-dragShouldLockBarVisibility| returns NO iff the bar is
651 TEST_F(BookmarkBarControllerTest, TestDragShouldLockBarVisibility) {
652 [bar_ updateState:BookmarkBar::HIDDEN
653 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
654 EXPECT_TRUE([bar_ dragShouldLockBarVisibility]);
656 [bar_ updateState:BookmarkBar::SHOW
657 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
658 EXPECT_TRUE([bar_ dragShouldLockBarVisibility]);
660 [bar_ updateState:BookmarkBar::DETACHED
661 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
662 EXPECT_FALSE([bar_ dragShouldLockBarVisibility]);
665 TEST_F(BookmarkBarControllerTest, TagMap) {
666 int64 ids[] = { 1, 3, 4, 40, 400, 4000, 800000000, 2, 123456789 };
667 std::vector<int32> tags;
669 // Generate some tags
670 for (unsigned int i = 0; i < arraysize(ids); i++) {
671 tags.push_back([bar_ menuTagFromNodeId:ids[i]]);
674 // Confirm reverse mapping.
675 for (unsigned int i = 0; i < arraysize(ids); i++) {
676 EXPECT_EQ(ids[i], [bar_ nodeIdFromMenuTag:tags[i]]);
679 // Confirm uniqueness.
680 std::sort(tags.begin(), tags.end());
681 for (unsigned int i=0; i<(tags.size()-1); i++) {
682 EXPECT_NE(tags[i], tags[i+1]);
686 TEST_F(BookmarkBarControllerTest, MenuForFolderNode) {
687 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
689 // First make sure something (e.g. "(empty)" string) is always present.
690 NSMenu* menu = [bar_ menuForFolderNode:model->bookmark_bar_node()];
691 EXPECT_GT([menu numberOfItems], 0);
693 // Test two bookmarks.
694 GURL gurl("http://www.foo.com");
695 bookmarks::AddIfNotBookmarked(model, gurl, ASCIIToUTF16("small"));
696 bookmarks::AddIfNotBookmarked(
697 model, GURL("http://www.cnn.com"), ASCIIToUTF16("bigger title"));
698 menu = [bar_ menuForFolderNode:model->bookmark_bar_node()];
699 EXPECT_EQ([menu numberOfItems], 2);
700 NSMenuItem *item = [menu itemWithTitle:@"bigger title"];
702 item = [menu itemWithTitle:@"small"];
705 int64 tag = [bar_ nodeIdFromMenuTag:[item tag]];
706 const BookmarkNode* node = bookmarks::GetBookmarkNodeByID(model, tag);
708 EXPECT_EQ(gurl, node->url());
711 // Test with an actual folder as well
712 const BookmarkNode* parent = model->bookmark_bar_node();
713 const BookmarkNode* folder = model->AddFolder(parent,
714 parent->child_count(),
715 ASCIIToUTF16("folder"));
716 model->AddURL(folder, folder->child_count(),
717 ASCIIToUTF16("f1"), GURL("http://framma-lamma.com"));
718 model->AddURL(folder, folder->child_count(),
719 ASCIIToUTF16("f2"), GURL("http://framma-lamma-ding-dong.com"));
720 menu = [bar_ menuForFolderNode:model->bookmark_bar_node()];
721 EXPECT_EQ([menu numberOfItems], 3);
723 item = [menu itemWithTitle:@"folder"];
725 EXPECT_TRUE([item hasSubmenu]);
726 NSMenu *submenu = [item submenu];
727 EXPECT_TRUE(submenu);
728 EXPECT_EQ(2, [submenu numberOfItems]);
729 EXPECT_TRUE([submenu itemWithTitle:@"f1"]);
730 EXPECT_TRUE([submenu itemWithTitle:@"f2"]);
733 // Confirm openBookmark: forwards the request to the controller's delegate
734 TEST_F(BookmarkBarControllerTest, OpenBookmark) {
735 GURL gurl("http://walla.walla.ding.dong.com");
736 scoped_ptr<BookmarkNode> node(new BookmarkNode(gurl));
738 base::scoped_nsobject<BookmarkButtonCell> cell(
739 [[BookmarkButtonCell alloc] init]);
740 [cell setBookmarkNode:node.get()];
741 base::scoped_nsobject<BookmarkButton> button([[BookmarkButton alloc] init]);
742 [button setCell:cell.get()];
743 [cell setRepresentedObject:[NSValue valueWithPointer:node.get()]];
745 [bar_ openBookmark:button];
746 EXPECT_EQ(noOpenBar()->urls_[0], node->url());
747 EXPECT_EQ(noOpenBar()->dispositions_[0], CURRENT_TAB);
750 TEST_F(BookmarkBarControllerTest, TestAddRemoveAndClear) {
751 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
752 NSView* buttonView = [bar_ buttonView];
753 EXPECT_EQ(0U, [[bar_ buttons] count]);
754 unsigned int initial_subview_count = [[buttonView subviews] count];
756 // Make sure a redundant call doesn't choke
757 [bar_ clearBookmarkBar];
758 EXPECT_EQ(0U, [[bar_ buttons] count]);
759 EXPECT_EQ(initial_subview_count, [[buttonView subviews] count]);
761 GURL gurl1("http://superfriends.hall-of-justice.edu");
762 // Short titles increase the chances of this test succeeding if the view is
764 // TODO(viettrungluu): make the test independent of window/view size, font
765 // metrics, button size and spacing, and everything else.
766 base::string16 title1(ASCIIToUTF16("x"));
767 bookmarks::AddIfNotBookmarked(model, gurl1, title1);
768 EXPECT_EQ(1U, [[bar_ buttons] count]);
769 EXPECT_EQ(1+initial_subview_count, [[buttonView subviews] count]);
771 GURL gurl2("http://legion-of-doom.gov");
772 base::string16 title2(ASCIIToUTF16("y"));
773 bookmarks::AddIfNotBookmarked(model, gurl2, title2);
774 EXPECT_EQ(2U, [[bar_ buttons] count]);
775 EXPECT_EQ(2+initial_subview_count, [[buttonView subviews] count]);
777 for (int i = 0; i < 3; i++) {
778 bookmarks::RemoveAllBookmarks(model, gurl2);
779 EXPECT_EQ(1U, [[bar_ buttons] count]);
780 EXPECT_EQ(1+initial_subview_count, [[buttonView subviews] count]);
783 bookmarks::AddIfNotBookmarked(model, gurl2, title2);
784 EXPECT_EQ(2U, [[bar_ buttons] count]);
785 EXPECT_EQ(2+initial_subview_count, [[buttonView subviews] count]);
788 [bar_ clearBookmarkBar];
789 EXPECT_EQ(0U, [[bar_ buttons] count]);
790 EXPECT_EQ(initial_subview_count, [[buttonView subviews] count]);
792 // Explicit test of loaded: since this is a convenient spot
794 EXPECT_EQ(2U, [[bar_ buttons] count]);
795 EXPECT_EQ(2+initial_subview_count, [[buttonView subviews] count]);
798 // Make sure we don't create too many buttons; we only really need
799 // ones that will be visible.
800 TEST_F(BookmarkBarControllerTest, TestButtonLimits) {
801 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
802 EXPECT_EQ(0U, [[bar_ buttons] count]);
803 // Add one; make sure we see it.
804 const BookmarkNode* parent = model->bookmark_bar_node();
805 model->AddURL(parent, parent->child_count(),
806 ASCIIToUTF16("title"), GURL("http://www.google.com"));
807 EXPECT_EQ(1U, [[bar_ buttons] count]);
809 // Add 30 which we expect to be 'too many'. Make sure we don't see
811 model->Remove(parent, 0);
812 EXPECT_EQ(0U, [[bar_ buttons] count]);
813 for (int i=0; i<30; i++) {
814 model->AddURL(parent, parent->child_count(),
815 ASCIIToUTF16("title"), GURL("http://www.google.com"));
817 int count = [[bar_ buttons] count];
818 EXPECT_LT(count, 30L);
820 // Add 10 more (to the front of the list so the on-screen buttons
821 // would change) and make sure the count stays the same.
822 for (int i=0; i<10; i++) {
823 model->AddURL(parent, 0, /* index is 0, so front, not end */
824 ASCIIToUTF16("title"), GURL("http://www.google.com"));
827 // Finally, grow the view and make sure the button count goes up.
828 NSRect frame = [[bar_ view] frame];
829 frame.size.width += 600;
830 [[bar_ view] setFrame:frame];
831 int finalcount = [[bar_ buttons] count];
832 EXPECT_GT(finalcount, count);
835 // Make sure that each button we add marches to the right and does not
836 // overlap with the previous one.
837 TEST_F(BookmarkBarControllerTest, TestButtonMarch) {
838 base::scoped_nsobject<NSMutableArray> cells([[NSMutableArray alloc] init]);
840 CGFloat widths[] = { 10, 10, 100, 10, 500, 500, 80000, 60000, 1, 345 };
841 for (unsigned int i = 0; i < arraysize(widths); i++) {
842 NSCell* cell = [[CellWithDesiredSize alloc]
844 desiredSize:NSMakeSize(widths[i], 30)];
845 [cells addObject:cell];
850 CGFloat x_end = x_offset; // end of the previous button
851 for (unsigned int i = 0; i < arraysize(widths); i++) {
852 NSRect r = [bar_ frameForBookmarkButtonFromCell:[cells objectAtIndex:i]
854 EXPECT_GE(r.origin.x, x_end);
859 TEST_F(BookmarkBarControllerTest, CheckForGrowth) {
860 WithNoAnimation at_all; // Turn off Cocoa auto animation in this scope.
861 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
862 GURL gurl1("http://www.google.com");
863 base::string16 title1(ASCIIToUTF16("x"));
864 bookmarks::AddIfNotBookmarked(model, gurl1, title1);
866 GURL gurl2("http://www.google.com/blah");
867 base::string16 title2(ASCIIToUTF16("y"));
868 bookmarks::AddIfNotBookmarked(model, gurl2, title2);
870 EXPECT_EQ(2U, [[bar_ buttons] count]);
871 CGFloat width_1 = [[[bar_ buttons] objectAtIndex:0] frame].size.width;
872 CGFloat x_2 = [[[bar_ buttons] objectAtIndex:1] frame].origin.x;
874 NSButton* first = [[bar_ buttons] objectAtIndex:0];
875 [[first cell] setTitle:@"This is a really big title; watch out mom!"];
876 [bar_ checkForBookmarkButtonGrowth:first];
878 // Make sure the 1st button is now wider, the 2nd one is moved over,
879 // and they don't overlap.
880 NSRect frame_1 = [[[bar_ buttons] objectAtIndex:0] frame];
881 NSRect frame_2 = [[[bar_ buttons] objectAtIndex:1] frame];
882 EXPECT_GT(frame_1.size.width, width_1);
883 EXPECT_GT(frame_2.origin.x, x_2);
884 EXPECT_GE(frame_2.origin.x, frame_1.origin.x + frame_1.size.width);
887 TEST_F(BookmarkBarControllerTest, DeleteBookmark) {
888 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
890 const char* urls[] = { "https://secret.url.com",
891 "http://super.duper.web.site.for.doodz.gov",
892 "http://www.foo-bar-baz.com/" };
893 const BookmarkNode* parent = model->bookmark_bar_node();
894 for (unsigned int i = 0; i < arraysize(urls); i++) {
895 model->AddURL(parent, parent->child_count(),
896 ASCIIToUTF16("title"), GURL(urls[i]));
898 EXPECT_EQ(3, parent->child_count());
899 const BookmarkNode* middle_node = parent->GetChild(1);
900 model->Remove(middle_node->parent(),
901 middle_node->parent()->GetIndexOf(middle_node));
903 EXPECT_EQ(2, parent->child_count());
904 EXPECT_EQ(parent->GetChild(0)->url(), GURL(urls[0]));
905 // node 2 moved into spot 1
906 EXPECT_EQ(parent->GetChild(1)->url(), GURL(urls[2]));
909 // TODO(jrg): write a test to confirm that nodeFaviconLoaded calls
910 // checkForBookmarkButtonGrowth:.
912 TEST_F(BookmarkBarControllerTest, Cell) {
913 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
916 const BookmarkNode* parent = model->bookmark_bar_node();
917 model->AddURL(parent, parent->child_count(),
918 ASCIIToUTF16("supertitle"),
919 GURL("http://superfriends.hall-of-justice.edu"));
920 const BookmarkNode* node = parent->GetChild(0);
922 NSCell* cell = [bar_ cellForBookmarkNode:node];
924 EXPECT_NSEQ(@"supertitle", [cell title]);
925 EXPECT_EQ(node, [[cell representedObject] pointerValue]);
926 EXPECT_TRUE([cell menu]);
928 // Empty cells still have a menu.
929 cell = [bar_ cellForBookmarkNode:nil];
930 EXPECT_TRUE([cell menu]);
931 // Even empty cells have a title (of "(empty)")
932 EXPECT_TRUE([cell title]);
934 // cell is autoreleased; no need to release here
937 // Test drawing, mostly to ensure nothing leaks or crashes.
938 TEST_F(BookmarkBarControllerTest, Display) {
939 [[bar_ view] display];
942 // Test that middle clicking on a bookmark button results in an open action.
943 TEST_F(BookmarkBarControllerTest, MiddleClick) {
944 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
945 GURL gurl1("http://www.google.com/");
946 base::string16 title1(ASCIIToUTF16("x"));
947 bookmarks::AddIfNotBookmarked(model, gurl1, title1);
949 EXPECT_EQ(1U, [[bar_ buttons] count]);
950 NSButton* first = [[bar_ buttons] objectAtIndex:0];
954 cocoa_test_event_utils::MouseEventWithType(NSOtherMouseUp, 0)];
955 EXPECT_EQ(noOpenBar()->urls_.size(), 1U);
958 TEST_F(BookmarkBarControllerTest, DisplaysHelpMessageOnEmpty) {
959 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
961 EXPECT_FALSE([[[bar_ buttonView] noItemContainer] isHidden]);
964 TEST_F(BookmarkBarControllerTest, HidesHelpMessageWithBookmark) {
965 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
967 const BookmarkNode* parent = model->bookmark_bar_node();
968 model->AddURL(parent, parent->child_count(),
969 ASCIIToUTF16("title"), GURL("http://one.com"));
972 EXPECT_TRUE([[[bar_ buttonView] noItemContainer] isHidden]);
975 TEST_F(BookmarkBarControllerTest, BookmarkButtonSizing) {
976 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
978 const BookmarkNode* parent = model->bookmark_bar_node();
979 model->AddURL(parent, parent->child_count(),
980 ASCIIToUTF16("title"), GURL("http://one.com"));
984 // Make sure the internal bookmark button also is the correct height.
985 NSArray* buttons = [bar_ buttons];
986 EXPECT_GT([buttons count], 0u);
987 for (NSButton* button in buttons) {
989 (chrome::kBookmarkBarHeight + bookmarks::kVisualHeightOffset) -
990 2 * bookmarks::kBookmarkVerticalPadding,
991 [button frame].size.height);
995 TEST_F(BookmarkBarControllerTest, DropBookmarks) {
996 const char* urls[] = {
999 "javascript:alert('lolwut')",
1000 "file://localhost/tmp/local-file.txt" // As if dragged from the desktop.
1002 const char* titles[] = {
1008 EXPECT_EQ(arraysize(urls), arraysize(titles));
1010 NSMutableArray* nsurls = [NSMutableArray array];
1011 NSMutableArray* nstitles = [NSMutableArray array];
1012 for (size_t i = 0; i < arraysize(urls); ++i) {
1013 [nsurls addObject:base::SysUTF8ToNSString(urls[i])];
1014 [nstitles addObject:base::SysUTF8ToNSString(titles[i])];
1017 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1018 const BookmarkNode* parent = model->bookmark_bar_node();
1019 [bar_ addURLs:nsurls withTitles:nstitles at:NSZeroPoint];
1020 EXPECT_EQ(4, parent->child_count());
1021 for (int i = 0; i < parent->child_count(); ++i) {
1022 GURL gurl = parent->GetChild(i)->url();
1023 if (gurl.scheme() == "http" ||
1024 gurl.scheme() == "javascript") {
1025 EXPECT_EQ(parent->GetChild(i)->url(), GURL(urls[i]));
1027 // Be flexible if the scheme needed to be added.
1028 std::string gurl_string = gurl.spec();
1029 std::string my_string = parent->GetChild(i)->url().spec();
1030 EXPECT_NE(gurl_string.find(my_string), std::string::npos);
1032 EXPECT_EQ(parent->GetChild(i)->GetTitle(), ASCIIToUTF16(titles[i]));
1036 TEST_F(BookmarkBarControllerTest, TestDragButton) {
1037 WithNoAnimation at_all;
1038 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1040 GURL gurls[] = { GURL("http://www.google.com/a"),
1041 GURL("http://www.google.com/b"),
1042 GURL("http://www.google.com/c") };
1043 base::string16 titles[] = { ASCIIToUTF16("a"),
1045 ASCIIToUTF16("c") };
1046 for (unsigned i = 0; i < arraysize(titles); i++)
1047 bookmarks::AddIfNotBookmarked(model, gurls[i], titles[i]);
1049 EXPECT_EQ([[bar_ buttons] count], arraysize(titles));
1050 EXPECT_NSEQ(@"a", [[[bar_ buttons] objectAtIndex:0] title]);
1052 [bar_ dragButton:[[bar_ buttons] objectAtIndex:2]
1055 EXPECT_NSEQ(@"c", [[[bar_ buttons] objectAtIndex:0] title]);
1056 // Make sure a 'copy' did not happen.
1057 EXPECT_EQ([[bar_ buttons] count], arraysize(titles));
1059 [bar_ dragButton:[[bar_ buttons] objectAtIndex:1]
1060 to:NSMakePoint(1000, 0)
1062 EXPECT_NSEQ(@"c", [[[bar_ buttons] objectAtIndex:0] title]);
1063 EXPECT_NSEQ(@"b", [[[bar_ buttons] objectAtIndex:1] title]);
1064 EXPECT_NSEQ(@"a", [[[bar_ buttons] objectAtIndex:2] title]);
1065 EXPECT_EQ([[bar_ buttons] count], arraysize(titles));
1067 // A drop of the 1st between the next 2.
1068 CGFloat x = NSMinX([[[bar_ buttons] objectAtIndex:2] frame]);
1069 x += [[bar_ view] frame].origin.x;
1070 [bar_ dragButton:[[bar_ buttons] objectAtIndex:0]
1071 to:NSMakePoint(x, 0)
1073 EXPECT_NSEQ(@"b", [[[bar_ buttons] objectAtIndex:0] title]);
1074 EXPECT_NSEQ(@"c", [[[bar_ buttons] objectAtIndex:1] title]);
1075 EXPECT_NSEQ(@"a", [[[bar_ buttons] objectAtIndex:2] title]);
1076 EXPECT_EQ([[bar_ buttons] count], arraysize(titles));
1078 // A drop on a non-folder button. (Shouldn't try and go in it.)
1079 x = NSMidX([[[bar_ buttons] objectAtIndex:0] frame]);
1080 x += [[bar_ view] frame].origin.x;
1081 [bar_ dragButton:[[bar_ buttons] objectAtIndex:2]
1082 to:NSMakePoint(x, 0)
1084 EXPECT_EQ(arraysize(titles), [[bar_ buttons] count]);
1086 // A drop on a folder button.
1087 const BookmarkNode* folder = model->AddFolder(
1088 model->bookmark_bar_node(), 0, ASCIIToUTF16("awesome folder"));
1090 model->AddURL(folder, 0, ASCIIToUTF16("already"),
1091 GURL("http://www.google.com"));
1092 EXPECT_EQ(arraysize(titles) + 1, [[bar_ buttons] count]);
1093 EXPECT_EQ(1, folder->child_count());
1094 x = NSMidX([[[bar_ buttons] objectAtIndex:0] frame]);
1095 x += [[bar_ view] frame].origin.x;
1096 base::string16 title =
1097 [[[bar_ buttons] objectAtIndex:2] bookmarkNode]->GetTitle();
1098 [bar_ dragButton:[[bar_ buttons] objectAtIndex:2]
1099 to:NSMakePoint(x, 0)
1101 // Gone from the bar
1102 EXPECT_EQ(arraysize(titles), [[bar_ buttons] count]);
1104 EXPECT_EQ(2, folder->child_count());
1106 EXPECT_EQ(title, folder->GetChild(1)->GetTitle());
1109 TEST_F(BookmarkBarControllerTest, TestCopyButton) {
1110 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1112 GURL gurls[] = { GURL("http://www.google.com/a"),
1113 GURL("http://www.google.com/b"),
1114 GURL("http://www.google.com/c") };
1115 base::string16 titles[] = { ASCIIToUTF16("a"),
1117 ASCIIToUTF16("c") };
1118 for (unsigned i = 0; i < arraysize(titles); i++)
1119 bookmarks::AddIfNotBookmarked(model, gurls[i], titles[i]);
1121 EXPECT_EQ([[bar_ buttons] count], arraysize(titles));
1122 EXPECT_NSEQ(@"a", [[[bar_ buttons] objectAtIndex:0] title]);
1124 // Drag 'a' between 'b' and 'c'.
1125 CGFloat x = NSMinX([[[bar_ buttons] objectAtIndex:2] frame]);
1126 x += [[bar_ view] frame].origin.x;
1127 [bar_ dragButton:[[bar_ buttons] objectAtIndex:0]
1128 to:NSMakePoint(x, 0)
1130 EXPECT_NSEQ(@"a", [[[bar_ buttons] objectAtIndex:0] title]);
1131 EXPECT_NSEQ(@"b", [[[bar_ buttons] objectAtIndex:1] title]);
1132 EXPECT_NSEQ(@"a", [[[bar_ buttons] objectAtIndex:2] title]);
1133 EXPECT_NSEQ(@"c", [[[bar_ buttons] objectAtIndex:3] title]);
1134 EXPECT_EQ([[bar_ buttons] count], 4U);
1137 // Fake a theme with colored text. Apply it and make sure bookmark
1138 // buttons have the same colored text. Repeat more than once.
1139 TEST_F(BookmarkBarControllerTest, TestThemedButton) {
1140 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1141 bookmarks::AddIfNotBookmarked(
1142 model, GURL("http://www.foo.com"), ASCIIToUTF16("small"));
1143 BookmarkButton* button = [[bar_ buttons] objectAtIndex:0];
1144 EXPECT_TRUE(button);
1146 NSArray* colors = [NSArray arrayWithObjects:[NSColor redColor],
1147 [NSColor blueColor],
1149 for (NSColor* color in colors) {
1150 FakeTheme theme(color);
1151 [bar_ updateTheme:&theme];
1152 NSAttributedString* astr = [button attributedTitle];
1154 EXPECT_NSEQ(@"small", [astr string]);
1155 // Pick a char in the middle to test (index 3)
1156 NSDictionary* attributes = [astr attributesAtIndex:3 effectiveRange:NULL];
1158 [attributes objectForKey:NSForegroundColorAttributeName];
1159 EXPECT_NSEQ(newColor, color);
1163 // Test that delegates and targets of buttons are cleared on dealloc.
1164 TEST_F(BookmarkBarControllerTest, TestClearOnDealloc) {
1165 // Make some bookmark buttons.
1166 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1167 GURL gurls[] = { GURL("http://www.foo.com/"),
1168 GURL("http://www.bar.com/"),
1169 GURL("http://www.baz.com/") };
1170 base::string16 titles[] = { ASCIIToUTF16("a"),
1172 ASCIIToUTF16("c") };
1173 for (size_t i = 0; i < arraysize(titles); i++)
1174 bookmarks::AddIfNotBookmarked(model, gurls[i], titles[i]);
1176 // Get and retain the buttons so we can examine them after dealloc.
1177 base::scoped_nsobject<NSArray> buttons([[bar_ buttons] retain]);
1178 EXPECT_EQ([buttons count], arraysize(titles));
1180 // Make sure that everything is set.
1181 for (BookmarkButton* button in buttons.get()) {
1182 ASSERT_TRUE([button isKindOfClass:[BookmarkButton class]]);
1183 EXPECT_TRUE([button delegate]);
1184 EXPECT_TRUE([button target]);
1185 EXPECT_TRUE([button action]);
1188 // This will dealloc....
1191 // Make sure that everything is cleared.
1192 for (BookmarkButton* button in buttons.get()) {
1193 EXPECT_FALSE([button delegate]);
1194 EXPECT_FALSE([button target]);
1195 EXPECT_FALSE([button action]);
1199 TEST_F(BookmarkBarControllerTest, TestFolders) {
1200 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1202 // Create some folder buttons.
1203 const BookmarkNode* parent = model->bookmark_bar_node();
1204 const BookmarkNode* folder = model->AddFolder(parent,
1205 parent->child_count(),
1206 ASCIIToUTF16("folder"));
1207 model->AddURL(folder, folder->child_count(),
1208 ASCIIToUTF16("f1"), GURL("http://framma-lamma.com"));
1209 folder = model->AddFolder(parent, parent->child_count(),
1210 ASCIIToUTF16("empty"));
1212 EXPECT_EQ([[bar_ buttons] count], 2U);
1214 // First confirm mouseEntered does nothing if "menus" aren't active.
1216 cocoa_test_event_utils::MouseEventWithType(NSOtherMouseUp, 0);
1217 [bar_ mouseEnteredButton:[[bar_ buttons] objectAtIndex:0] event:event];
1218 EXPECT_FALSE([bar_ folderController]);
1220 // Make one active. Entering it is now a no-op.
1221 [bar_ openBookmarkFolderFromButton:[[bar_ buttons] objectAtIndex:0]];
1222 BookmarkBarFolderController* bbfc = [bar_ folderController];
1224 [bar_ mouseEnteredButton:[[bar_ buttons] objectAtIndex:0] event:event];
1225 EXPECT_EQ(bbfc, [bar_ folderController]);
1227 // Enter a different one; a new folderController is active.
1228 [bar_ mouseEnteredButton:[[bar_ buttons] objectAtIndex:1] event:event];
1229 EXPECT_NE(bbfc, [bar_ folderController]);
1231 // Confirm exited is a no-op.
1232 [bar_ mouseExitedButton:[[bar_ buttons] objectAtIndex:1] event:event];
1233 EXPECT_NE(bbfc, [bar_ folderController]);
1236 [bar_ closeBookmarkFolder:nil];
1239 // Verify that the folder menu presentation properly tracks mouse movements
1240 // over the bar. Until there is a click no folder menus should show. After a
1241 // click on a folder folder menus should show until another click on a folder
1242 // button, and a click outside the bar and its folder menus.
1243 TEST_F(BookmarkBarControllerTest, TestFolderButtons) {
1244 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1245 const BookmarkNode* root = model->bookmark_bar_node();
1246 const std::string model_string("1b 2f:[ 2f1b 2f2b ] 3b 4f:[ 4f1b 4f2b ] ");
1247 test::AddNodesFromModelString(model, root, model_string);
1249 // Validate initial model and that we do not have a folder controller.
1250 std::string actualModelString = test::ModelStringFromNode(root);
1251 EXPECT_EQ(model_string, actualModelString);
1252 EXPECT_FALSE([bar_ folderController]);
1254 // Add a real bookmark so we can click on it.
1255 const BookmarkNode* folder = root->GetChild(3);
1256 model->AddURL(folder, folder->child_count(), ASCIIToUTF16("CLICK ME"),
1257 GURL("http://www.google.com/"));
1259 // Click on a folder button.
1260 BookmarkButton* button = [bar_ buttonWithTitleEqualTo:@"4f"];
1261 EXPECT_TRUE(button);
1262 [bar_ openBookmarkFolderFromButton:button];
1263 BookmarkBarFolderController* bbfc = [bar_ folderController];
1266 // Make sure a 2nd click on the same button closes things.
1267 [bar_ openBookmarkFolderFromButton:button];
1268 EXPECT_FALSE([bar_ folderController]);
1270 // Next open is a different button.
1271 button = [bar_ buttonWithTitleEqualTo:@"2f"];
1272 EXPECT_TRUE(button);
1273 [bar_ openBookmarkFolderFromButton:button];
1274 EXPECT_TRUE([bar_ folderController]);
1276 // Mouse over a non-folder button and confirm controller has gone away.
1277 button = [bar_ buttonWithTitleEqualTo:@"1b"];
1278 EXPECT_TRUE(button);
1279 NSEvent* event = cocoa_test_event_utils::MouseEventAtPoint([button center],
1281 [bar_ mouseEnteredButton:button event:event];
1282 EXPECT_FALSE([bar_ folderController]);
1284 // Mouse over the original folder and confirm a new controller.
1285 button = [bar_ buttonWithTitleEqualTo:@"2f"];
1286 EXPECT_TRUE(button);
1287 [bar_ mouseEnteredButton:button event:event];
1288 BookmarkBarFolderController* oldBBFC = [bar_ folderController];
1289 EXPECT_TRUE(oldBBFC);
1291 // 'Jump' over to a different folder and confirm a new controller.
1292 button = [bar_ buttonWithTitleEqualTo:@"4f"];
1293 EXPECT_TRUE(button);
1294 [bar_ mouseEnteredButton:button event:event];
1295 BookmarkBarFolderController* newBBFC = [bar_ folderController];
1296 EXPECT_TRUE(newBBFC);
1297 EXPECT_NE(oldBBFC, newBBFC);
1300 // Make sure the "off the side" folder looks like a bookmark folder
1301 // but only contains "off the side" items.
1302 TEST_F(BookmarkBarControllerTest, OffTheSideFolder) {
1304 // It starts hidden.
1305 EXPECT_TRUE([bar_ offTheSideButtonIsHidden]);
1307 // Create some buttons.
1308 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1309 const BookmarkNode* parent = model->bookmark_bar_node();
1310 for (int x = 0; x < 30; x++) {
1311 model->AddURL(parent, parent->child_count(),
1312 ASCIIToUTF16("medium-size-title"),
1313 GURL("http://framma-lamma.com"));
1315 // Add a couple more so we can delete one and make sure its button goes away.
1316 model->AddURL(parent, parent->child_count(),
1317 ASCIIToUTF16("DELETE_ME"), GURL("http://ashton-tate.com"));
1318 model->AddURL(parent, parent->child_count(),
1319 ASCIIToUTF16("medium-size-title"),
1320 GURL("http://framma-lamma.com"));
1322 // Should no longer be hidden.
1323 EXPECT_FALSE([bar_ offTheSideButtonIsHidden]);
1325 // Open it; make sure we have a folder controller.
1326 EXPECT_FALSE([bar_ folderController]);
1327 [bar_ openOffTheSideFolderFromButton:[bar_ offTheSideButton]];
1328 BookmarkBarFolderController* bbfc = [bar_ folderController];
1331 // Confirm the contents are only buttons which fell off the side by
1332 // making sure that none of the nodes in the off-the-side folder are
1333 // found in bar buttons. Be careful since not all the bar buttons
1334 // may be currently displayed.
1335 NSArray* folderButtons = [bbfc buttons];
1336 NSArray* barButtons = [bar_ buttons];
1337 for (BookmarkButton* folderButton in folderButtons) {
1338 for (BookmarkButton* barButton in barButtons) {
1339 if ([barButton superview]) {
1340 EXPECT_NE([folderButton bookmarkNode], [barButton bookmarkNode]);
1345 // Delete a bookmark in the off-the-side and verify it's gone.
1346 BookmarkButton* button = [bbfc buttonWithTitleEqualTo:@"DELETE_ME"];
1347 EXPECT_TRUE(button);
1348 model->Remove(parent, parent->child_count() - 2);
1349 button = [bbfc buttonWithTitleEqualTo:@"DELETE_ME"];
1350 EXPECT_FALSE(button);
1353 TEST_F(BookmarkBarControllerTest, EventToExitCheck) {
1354 NSEvent* event = cocoa_test_event_utils::MouseEventWithType(NSMouseMoved, 0);
1355 EXPECT_FALSE([bar_ isEventAnExitEvent:event]);
1357 BookmarkBarFolderWindow* folderWindow = [[[BookmarkBarFolderWindow alloc]
1359 [[[bar_ view] window] addChildWindow:folderWindow
1360 ordered:NSWindowAbove];
1361 event = cocoa_test_event_utils::LeftMouseDownAtPointInWindow(NSMakePoint(1,1),
1363 EXPECT_FALSE([bar_ isEventAnExitEvent:event]);
1365 event = cocoa_test_event_utils::LeftMouseDownAtPointInWindow(
1366 NSMakePoint(100,100), test_window());
1367 EXPECT_TRUE([bar_ isEventAnExitEvent:event]);
1369 // Many components are arbitrary (e.g. location, keycode).
1370 event = [NSEvent keyEventWithType:NSKeyDown
1371 location:NSMakePoint(1,1)
1377 charactersIgnoringModifiers:@"x"
1380 EXPECT_FALSE([bar_ isEventAnExitEvent:event]);
1382 [[[bar_ view] window] removeChildWindow:folderWindow];
1385 TEST_F(BookmarkBarControllerTest, DropDestination) {
1386 // Make some buttons.
1387 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1388 const BookmarkNode* parent = model->bookmark_bar_node();
1389 model->AddFolder(parent, parent->child_count(), ASCIIToUTF16("folder 1"));
1390 model->AddFolder(parent, parent->child_count(), ASCIIToUTF16("folder 2"));
1391 EXPECT_EQ([[bar_ buttons] count], 2U);
1393 // Confirm "off to left" and "off to right" match nothing.
1394 NSPoint p = NSMakePoint(-1, 2);
1395 EXPECT_FALSE([bar_ buttonForDroppingOnAtPoint:p]);
1396 EXPECT_TRUE([bar_ shouldShowIndicatorShownForPoint:p]);
1397 p = NSMakePoint(50000, 10);
1398 EXPECT_FALSE([bar_ buttonForDroppingOnAtPoint:p]);
1399 EXPECT_TRUE([bar_ shouldShowIndicatorShownForPoint:p]);
1401 // Confirm "right in the center" (give or take a pixel) is a match,
1402 // and confirm "just barely in the button" is not. Anything more
1403 // specific seems likely to be tweaked.
1404 CGFloat viewFrameXOffset = [[bar_ view] frame].origin.x;
1405 for (BookmarkButton* button in [bar_ buttons]) {
1406 CGFloat x = NSMidX([button frame]) + viewFrameXOffset;
1407 // Somewhere near the center: a match
1409 [bar_ buttonForDroppingOnAtPoint:NSMakePoint(x-1, 10)]);
1411 [bar_ buttonForDroppingOnAtPoint:NSMakePoint(x+1, 10)]);
1412 EXPECT_FALSE([bar_ shouldShowIndicatorShownForPoint:NSMakePoint(x, 10)]);;
1414 // On the very edges: NOT a match
1415 x = NSMinX([button frame]) + viewFrameXOffset;
1417 [bar_ buttonForDroppingOnAtPoint:NSMakePoint(x, 9)]);
1418 x = NSMaxX([button frame]) + viewFrameXOffset;
1420 [bar_ buttonForDroppingOnAtPoint:NSMakePoint(x, 11)]);
1424 TEST_F(BookmarkBarControllerTest, CloseFolderOnAnimate) {
1425 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1426 [bar_ setStateAnimationsEnabled:YES];
1427 const BookmarkNode* parent = model->bookmark_bar_node();
1428 const BookmarkNode* folder = model->AddFolder(parent,
1429 parent->child_count(),
1430 ASCIIToUTF16("folder"));
1431 model->AddFolder(parent, parent->child_count(),
1432 ASCIIToUTF16("sibbling folder"));
1433 model->AddURL(folder, folder->child_count(), ASCIIToUTF16("title a"),
1434 GURL("http://www.google.com/a"));
1435 model->AddURL(folder, folder->child_count(),
1436 ASCIIToUTF16("title super duper long long whoa momma title you betcha"),
1437 GURL("http://www.google.com/b"));
1438 BookmarkButton* button = [[bar_ buttons] objectAtIndex:0];
1439 EXPECT_FALSE([bar_ folderController]);
1440 [bar_ openBookmarkFolderFromButton:button];
1441 BookmarkBarFolderController* bbfc = [bar_ folderController];
1442 // The following tells us that the folder menu is showing. We want to make
1443 // sure the folder menu goes away if the bookmark bar is hidden.
1445 EXPECT_TRUE([bar_ isVisible]);
1447 // Hide the bookmark bar.
1448 [bar_ updateState:BookmarkBar::DETACHED
1449 changeType:BookmarkBar::ANIMATE_STATE_CHANGE];
1450 EXPECT_TRUE([bar_ isAnimationRunning]);
1452 // Now that we've closed the bookmark bar (with animation) the folder menu
1453 // should have been closed thus releasing the folderController.
1454 EXPECT_FALSE([bar_ folderController]);
1456 // Stop the pending animation to tear down cleanly.
1457 [bar_ updateState:BookmarkBar::DETACHED
1458 changeType:BookmarkBar::DONT_ANIMATE_STATE_CHANGE];
1459 EXPECT_FALSE([bar_ isAnimationRunning]);
1462 TEST_F(BookmarkBarControllerTest, MoveRemoveAddButtons) {
1463 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1464 const BookmarkNode* root = model->bookmark_bar_node();
1465 const std::string model_string("1b 2f:[ 2f1b 2f2b ] 3b ");
1466 test::AddNodesFromModelString(model, root, model_string);
1468 // Validate initial model.
1469 std::string actualModelString = test::ModelStringFromNode(root);
1470 EXPECT_EQ(model_string, actualModelString);
1472 // Remember how many buttons are showing.
1473 int oldDisplayedButtons = [bar_ displayedButtonCount];
1474 NSArray* buttons = [bar_ buttons];
1476 // Move a button around a bit.
1477 [bar_ moveButtonFromIndex:0 toIndex:2];
1478 EXPECT_NSEQ(@"2f", [[buttons objectAtIndex:0] title]);
1479 EXPECT_NSEQ(@"3b", [[buttons objectAtIndex:1] title]);
1480 EXPECT_NSEQ(@"1b", [[buttons objectAtIndex:2] title]);
1481 EXPECT_EQ(oldDisplayedButtons, [bar_ displayedButtonCount]);
1482 [bar_ moveButtonFromIndex:2 toIndex:0];
1483 EXPECT_NSEQ(@"1b", [[buttons objectAtIndex:0] title]);
1484 EXPECT_NSEQ(@"2f", [[buttons objectAtIndex:1] title]);
1485 EXPECT_NSEQ(@"3b", [[buttons objectAtIndex:2] title]);
1486 EXPECT_EQ(oldDisplayedButtons, [bar_ displayedButtonCount]);
1488 // Add a couple of buttons.
1489 const BookmarkNode* parent = root->GetChild(1); // Purloin an existing node.
1490 const BookmarkNode* node = parent->GetChild(0);
1491 [bar_ addButtonForNode:node atIndex:0];
1492 EXPECT_NSEQ(@"2f1b", [[buttons objectAtIndex:0] title]);
1493 EXPECT_NSEQ(@"1b", [[buttons objectAtIndex:1] title]);
1494 EXPECT_NSEQ(@"2f", [[buttons objectAtIndex:2] title]);
1495 EXPECT_NSEQ(@"3b", [[buttons objectAtIndex:3] title]);
1496 EXPECT_EQ(oldDisplayedButtons + 1, [bar_ displayedButtonCount]);
1497 node = parent->GetChild(1);
1498 [bar_ addButtonForNode:node atIndex:-1];
1499 EXPECT_NSEQ(@"2f1b", [[buttons objectAtIndex:0] title]);
1500 EXPECT_NSEQ(@"1b", [[buttons objectAtIndex:1] title]);
1501 EXPECT_NSEQ(@"2f", [[buttons objectAtIndex:2] title]);
1502 EXPECT_NSEQ(@"3b", [[buttons objectAtIndex:3] title]);
1503 EXPECT_NSEQ(@"2f2b", [[buttons objectAtIndex:4] title]);
1504 EXPECT_EQ(oldDisplayedButtons + 2, [bar_ displayedButtonCount]);
1506 // Remove a couple of buttons.
1507 [bar_ removeButton:4 animate:NO];
1508 [bar_ removeButton:1 animate:NO];
1509 EXPECT_NSEQ(@"2f1b", [[buttons objectAtIndex:0] title]);
1510 EXPECT_NSEQ(@"2f", [[buttons objectAtIndex:1] title]);
1511 EXPECT_NSEQ(@"3b", [[buttons objectAtIndex:2] title]);
1512 EXPECT_EQ(oldDisplayedButtons, [bar_ displayedButtonCount]);
1515 TEST_F(BookmarkBarControllerTest, ShrinkOrHideView) {
1516 NSRect viewFrame = NSMakeRect(0.0, 0.0, 500.0, 50.0);
1517 NSView* view = [[[NSView alloc] initWithFrame:viewFrame] autorelease];
1518 EXPECT_FALSE([view isHidden]);
1519 [bar_ shrinkOrHideView:view forMaxX:500.0];
1520 EXPECT_EQ(500.0, NSWidth([view frame]));
1521 EXPECT_FALSE([view isHidden]);
1522 [bar_ shrinkOrHideView:view forMaxX:450.0];
1523 EXPECT_EQ(450.0, NSWidth([view frame]));
1524 EXPECT_FALSE([view isHidden]);
1525 [bar_ shrinkOrHideView:view forMaxX:40.0];
1526 EXPECT_EQ(40.0, NSWidth([view frame]));
1527 EXPECT_FALSE([view isHidden]);
1528 [bar_ shrinkOrHideView:view forMaxX:31.0];
1529 EXPECT_EQ(31.0, NSWidth([view frame]));
1530 EXPECT_FALSE([view isHidden]);
1531 [bar_ shrinkOrHideView:view forMaxX:29.0];
1532 EXPECT_TRUE([view isHidden]);
1535 TEST_F(BookmarkBarControllerTest, LastBookmarkResizeBehavior) {
1536 // Hide the apps shortcut.
1537 profile()->GetPrefs()->SetBoolean(prefs::kShowAppsShortcutInBookmarkBar,
1539 ASSERT_TRUE([bar_ appsPageShortcutButtonIsHidden]);
1541 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1542 const BookmarkNode* root = model->bookmark_bar_node();
1543 const std::string model_string("1b 2f:[ 2f1b 2f2b ] 3b ");
1544 test::AddNodesFromModelString(model, root, model_string);
1545 [bar_ frameDidChange];
1547 CGFloat viewWidths[] = { 123.0, 124.0, 151.0, 152.0, 153.0, 154.0, 155.0,
1548 200.0, 155.0, 154.0, 153.0, 152.0, 151.0, 124.0,
1550 BOOL offTheSideButtonIsHiddenResults[] = { NO, NO, NO, NO, YES, YES, YES, YES,
1551 YES, YES, YES, NO, NO, NO, NO};
1552 int displayedButtonCountResults[] = { 1, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 2, 2,
1555 for (unsigned int i = 0; i < sizeof(viewWidths) / sizeof(viewWidths[0]);
1557 NSRect frame = [[bar_ view] frame];
1558 frame.size.width = viewWidths[i] + bookmarks::kBookmarkRightMargin;
1559 [[bar_ view] setFrame:frame];
1560 EXPECT_EQ(offTheSideButtonIsHiddenResults[i],
1561 [bar_ offTheSideButtonIsHidden]);
1562 EXPECT_EQ(displayedButtonCountResults[i], [bar_ displayedButtonCount]);
1566 TEST_F(BookmarkBarControllerTest, BookmarksWithAppsPageShortcut) {
1567 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1568 const BookmarkNode* root = model->bookmark_bar_node();
1569 const std::string model_string("1b 2f:[ 2f1b 2f2b ] 3b ");
1570 test::AddNodesFromModelString(model, root, model_string);
1571 [bar_ frameDidChange];
1573 // Apps page shortcut button should be visible.
1574 ASSERT_FALSE([bar_ appsPageShortcutButtonIsHidden]);
1576 // Bookmarks should be to the right of the Apps page shortcut button.
1577 CGFloat apps_button_right = NSMaxX([[bar_ appsPageShortcutButton] frame]);
1578 CGFloat right = apps_button_right;
1579 NSArray* buttons = [bar_ buttons];
1580 for (size_t i = 0; i < [buttons count]; ++i) {
1581 EXPECT_LE(right, NSMinX([[buttons objectAtIndex:i] frame]));
1582 right = NSMaxX([[buttons objectAtIndex:i] frame]);
1585 // Removing the Apps button should move every bookmark to the left.
1586 profile()->GetPrefs()->SetBoolean(prefs::kShowAppsShortcutInBookmarkBar,
1588 ASSERT_TRUE([bar_ appsPageShortcutButtonIsHidden]);
1589 EXPECT_GT(apps_button_right, NSMinX([[buttons objectAtIndex:0] frame]));
1590 for (size_t i = 1; i < [buttons count]; ++i) {
1591 EXPECT_LE(NSMaxX([[buttons objectAtIndex:i - 1] frame]),
1592 NSMinX([[buttons objectAtIndex:i] frame]));
1596 TEST_F(BookmarkBarControllerTest, BookmarksWithoutAppsPageShortcut) {
1597 // The no item containers should be to the right of the Apps button.
1598 ASSERT_FALSE([bar_ appsPageShortcutButtonIsHidden]);
1599 CGFloat apps_button_right = NSMaxX([[bar_ appsPageShortcutButton] frame]);
1600 EXPECT_LE(apps_button_right,
1601 NSMinX([[[bar_ buttonView] noItemTextfield] frame]));
1602 EXPECT_LE(NSMaxX([[[bar_ buttonView] noItemTextfield] frame]),
1603 NSMinX([[[bar_ buttonView] importBookmarksButton] frame]));
1605 // Removing the Apps button should move the no item containers to the left.
1606 profile()->GetPrefs()->SetBoolean(prefs::kShowAppsShortcutInBookmarkBar,
1608 ASSERT_TRUE([bar_ appsPageShortcutButtonIsHidden]);
1609 EXPECT_GT(apps_button_right,
1610 NSMinX([[[bar_ buttonView] noItemTextfield] frame]));
1611 EXPECT_LE(NSMaxX([[[bar_ buttonView] noItemTextfield] frame]),
1612 NSMinX([[[bar_ buttonView] importBookmarksButton] frame]));
1615 TEST_F(BookmarkBarControllerTest, ManagedShowAppsShortcutInBookmarksBar) {
1616 // By default the pref is not managed and the apps shortcut is shown.
1617 TestingPrefServiceSyncable* prefs = profile()->GetTestingPrefService();
1619 prefs->IsManagedPreference(prefs::kShowAppsShortcutInBookmarkBar));
1620 EXPECT_FALSE([bar_ appsPageShortcutButtonIsHidden]);
1622 // Hide the apps shortcut by policy, via the managed pref.
1623 prefs->SetManagedPref(prefs::kShowAppsShortcutInBookmarkBar,
1624 new base::FundamentalValue(false));
1625 EXPECT_TRUE([bar_ appsPageShortcutButtonIsHidden]);
1627 // And try showing it via policy too.
1628 prefs->SetManagedPref(prefs::kShowAppsShortcutInBookmarkBar,
1629 new base::FundamentalValue(true));
1630 EXPECT_FALSE([bar_ appsPageShortcutButtonIsHidden]);
1633 class BookmarkBarControllerOpenAllTest : public BookmarkBarControllerTest {
1635 virtual void SetUp() {
1636 BookmarkBarControllerTest::SetUp();
1637 ASSERT_TRUE(profile());
1639 resizeDelegate_.reset([[ViewResizerPong alloc] init]);
1640 NSRect parent_frame = NSMakeRect(0, 0, 800, 50);
1642 [[BookmarkBarControllerOpenAllPong alloc]
1643 initWithBrowser:browser()
1644 initialWidth:NSWidth(parent_frame)
1646 resizeDelegate:resizeDelegate_.get()]);
1648 // Awkwardness to look like we've been installed.
1649 [parent_view_ addSubview:[bar_ view]];
1650 NSRect frame = [[[bar_ view] superview] frame];
1651 frame.origin.y = 100;
1652 [[[bar_ view] superview] setFrame:frame];
1654 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1655 parent_ = model->bookmark_bar_node();
1656 // { one, { two-one, two-two }, three }
1657 model->AddURL(parent_, parent_->child_count(), ASCIIToUTF16("title"),
1658 GURL("http://one.com"));
1659 folder_ = model->AddFolder(parent_, parent_->child_count(),
1660 ASCIIToUTF16("folder"));
1661 model->AddURL(folder_, folder_->child_count(),
1662 ASCIIToUTF16("title"), GURL("http://two-one.com"));
1663 model->AddURL(folder_, folder_->child_count(),
1664 ASCIIToUTF16("title"), GURL("http://two-two.com"));
1665 model->AddURL(parent_, parent_->child_count(),
1666 ASCIIToUTF16("title"), GURL("https://three.com"));
1668 const BookmarkNode* parent_; // Weak
1669 const BookmarkNode* folder_; // Weak
1672 // Command-click on a folder should open all the bookmarks in it.
1673 TEST_F(BookmarkBarControllerOpenAllTest, CommandClickOnFolder) {
1674 NSButton* first = [[bar_ buttons] objectAtIndex:0];
1677 // Create the right kind of event; mock NSApp so [NSApp
1678 // currentEvent] finds it.
1679 NSEvent* commandClick =
1680 cocoa_test_event_utils::MouseEventAtPoint(NSZeroPoint,
1683 id fakeApp = [OCMockObject partialMockForObject:NSApp];
1684 [[[fakeApp stub] andReturn:commandClick] currentEvent];
1687 size_t originalDispositionCount = noOpenBar()->dispositions_.size();
1690 [first performClick:first];
1692 size_t dispositionCount = noOpenBar()->dispositions_.size();
1693 EXPECT_EQ(originalDispositionCount+1, dispositionCount);
1694 EXPECT_EQ(noOpenBar()->dispositions_[dispositionCount-1], NEW_BACKGROUND_TAB);
1700 class BookmarkBarControllerNotificationTest : public CocoaProfileTest {
1702 virtual void SetUp() {
1703 CocoaProfileTest::SetUp();
1704 ASSERT_TRUE(browser());
1706 resizeDelegate_.reset([[ViewResizerPong alloc] init]);
1707 NSRect parent_frame = NSMakeRect(0, 0, 800, 50);
1708 parent_view_.reset([[NSView alloc] initWithFrame:parent_frame]);
1709 [parent_view_ setHidden:YES];
1711 [[BookmarkBarControllerNotificationPong alloc]
1712 initWithBrowser:browser()
1713 initialWidth:NSWidth(parent_frame)
1715 resizeDelegate:resizeDelegate_.get()]);
1717 // Force loading of the nib.
1719 // Awkwardness to look like we've been installed.
1720 [parent_view_ addSubview:[bar_ view]];
1721 NSRect frame = [[[bar_ view] superview] frame];
1722 frame.origin.y = 100;
1723 [[[bar_ view] superview] setFrame:frame];
1725 // Do not add the bar to a window, yet.
1728 base::scoped_nsobject<NSView> parent_view_;
1729 base::scoped_nsobject<ViewResizerPong> resizeDelegate_;
1730 base::scoped_nsobject<BookmarkBarControllerNotificationPong> bar_;
1733 TEST_F(BookmarkBarControllerNotificationTest, DeregistersForNotifications) {
1734 NSWindow* window = [[CocoaTestHelperWindow alloc] init];
1735 [window setReleasedWhenClosed:YES];
1737 // First add the bookmark bar to the temp window, then to another window.
1738 [[window contentView] addSubview:parent_view_];
1739 [[test_window() contentView] addSubview:parent_view_];
1741 // Post a fake windowDidResignKey notification for the temp window and make
1742 // sure the bookmark bar controller wasn't listening.
1743 [[NSNotificationCenter defaultCenter]
1744 postNotificationName:NSWindowDidResignKeyNotification
1746 EXPECT_FALSE([bar_ windowDidResignKeyReceived]);
1748 // Close the temp window and make sure no notification was received.
1750 EXPECT_FALSE([bar_ windowWillCloseReceived]);
1754 // TODO(jrg): draggingEntered: and draggingExited: trigger timers so
1755 // they are hard to test. Factor out "fire timers" into routines
1756 // which can be overridden to fire immediately to make behavior
1759 // TODO(jrg): add unit test to make sure "Other Bookmarks" responds
1760 // properly to a hover open.
1762 // TODO(viettrungluu): figure out how to test animations.
1764 class BookmarkBarControllerDragDropTest : public BookmarkBarControllerTestBase {
1766 base::scoped_nsobject<BookmarkBarControllerDragData> bar_;
1768 virtual void SetUp() {
1769 BookmarkBarControllerTestBase::SetUp();
1770 ASSERT_TRUE(browser());
1773 [[BookmarkBarControllerDragData alloc]
1774 initWithBrowser:browser()
1775 initialWidth:NSWidth([parent_view_ frame])
1777 resizeDelegate:resizeDelegate_.get()]);
1778 InstallAndToggleBar(bar_.get());
1782 TEST_F(BookmarkBarControllerDragDropTest, DragMoveBarBookmarkToOffTheSide) {
1783 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1784 const BookmarkNode* root = model->bookmark_bar_node();
1785 const std::string model_string("1bWithLongName 2fWithLongName:[ "
1786 "2f1bWithLongName 2f2fWithLongName:[ 2f2f1bWithLongName "
1787 "2f2f2bWithLongName 2f2f3bWithLongName 2f4b ] 2f3bWithLongName ] "
1788 "3bWithLongName 4bWithLongName 5bWithLongName 6bWithLongName "
1789 "7bWithLongName 8bWithLongName 9bWithLongName 10bWithLongName "
1790 "11bWithLongName 12bWithLongName 13b ");
1791 test::AddNodesFromModelString(model, root, model_string);
1793 // Validate initial model.
1794 std::string actualModelString = test::ModelStringFromNode(root);
1795 EXPECT_EQ(model_string, actualModelString);
1797 // Insure that the off-the-side is not showing.
1798 ASSERT_FALSE([bar_ offTheSideButtonIsHidden]);
1800 // Remember how many buttons are showing and are available.
1801 int oldDisplayedButtons = [bar_ displayedButtonCount];
1802 int oldChildCount = root->child_count();
1804 // Pop up the off-the-side menu.
1805 BookmarkButton* otsButton = (BookmarkButton*)[bar_ offTheSideButton];
1806 ASSERT_TRUE(otsButton);
1807 [[otsButton target] performSelector:@selector(openOffTheSideFolderFromButton:)
1808 withObject:otsButton];
1809 BookmarkBarFolderController* otsController = [bar_ folderController];
1810 EXPECT_TRUE(otsController);
1811 NSWindow* toWindow = [otsController window];
1812 EXPECT_TRUE(toWindow);
1813 BookmarkButton* draggedButton =
1814 [bar_ buttonWithTitleEqualTo:@"3bWithLongName"];
1815 ASSERT_TRUE(draggedButton);
1816 int oldOTSCount = (int)[[otsController buttons] count];
1817 EXPECT_EQ(oldOTSCount, oldChildCount - oldDisplayedButtons);
1818 BookmarkButton* targetButton = [[otsController buttons] objectAtIndex:0];
1819 ASSERT_TRUE(targetButton);
1820 [otsController dragButton:draggedButton
1821 to:[targetButton center]
1823 // There should still be the same number of buttons in the bar
1824 // and off-the-side should have one more.
1825 int newDisplayedButtons = [bar_ displayedButtonCount];
1826 int newChildCount = root->child_count();
1827 int newOTSCount = (int)[[otsController buttons] count];
1828 EXPECT_EQ(oldDisplayedButtons, newDisplayedButtons);
1829 EXPECT_EQ(oldChildCount + 1, newChildCount);
1830 EXPECT_EQ(oldOTSCount + 1, newOTSCount);
1831 EXPECT_EQ(newOTSCount, newChildCount - newDisplayedButtons);
1834 TEST_F(BookmarkBarControllerDragDropTest, DragOffTheSideToOther) {
1835 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1836 const BookmarkNode* root = model->bookmark_bar_node();
1837 const std::string model_string("1bWithLongName 2bWithLongName "
1838 "3bWithLongName 4bWithLongName 5bWithLongName 6bWithLongName "
1839 "7bWithLongName 8bWithLongName 9bWithLongName 10bWithLongName "
1840 "11bWithLongName 12bWithLongName 13bWithLongName 14bWithLongName "
1841 "15bWithLongName 16bWithLongName 17bWithLongName 18bWithLongName "
1842 "19bWithLongName 20bWithLongName ");
1843 test::AddNodesFromModelString(model, root, model_string);
1845 const BookmarkNode* other = model->other_node();
1846 const std::string other_string("1other 2other 3other ");
1847 test::AddNodesFromModelString(model, other, other_string);
1849 // Validate initial model.
1850 std::string actualModelString = test::ModelStringFromNode(root);
1851 EXPECT_EQ(model_string, actualModelString);
1852 std::string actualOtherString = test::ModelStringFromNode(other);
1853 EXPECT_EQ(other_string, actualOtherString);
1855 // Insure that the off-the-side is showing.
1856 ASSERT_FALSE([bar_ offTheSideButtonIsHidden]);
1858 // Remember how many buttons are showing and are available.
1859 int oldDisplayedButtons = [bar_ displayedButtonCount];
1860 int oldRootCount = root->child_count();
1861 int oldOtherCount = other->child_count();
1863 // Pop up the off-the-side menu.
1864 BookmarkButton* otsButton = (BookmarkButton*)[bar_ offTheSideButton];
1865 ASSERT_TRUE(otsButton);
1866 [[otsButton target] performSelector:@selector(openOffTheSideFolderFromButton:)
1867 withObject:otsButton];
1868 BookmarkBarFolderController* otsController = [bar_ folderController];
1869 EXPECT_TRUE(otsController);
1870 int oldOTSCount = (int)[[otsController buttons] count];
1871 EXPECT_EQ(oldOTSCount, oldRootCount - oldDisplayedButtons);
1873 // Pick an off-the-side button and drag it to the other bookmarks.
1874 BookmarkButton* draggedButton =
1875 [otsController buttonWithTitleEqualTo:@"20bWithLongName"];
1876 ASSERT_TRUE(draggedButton);
1877 BookmarkButton* targetButton = [bar_ otherBookmarksButton];
1878 ASSERT_TRUE(targetButton);
1879 [bar_ dragButton:draggedButton to:[targetButton center] copy:NO];
1881 // There should one less button in the bar, one less in off-the-side,
1882 // and one more in other bookmarks.
1883 int newRootCount = root->child_count();
1884 int newOTSCount = (int)[[otsController buttons] count];
1885 int newOtherCount = other->child_count();
1886 EXPECT_EQ(oldRootCount - 1, newRootCount);
1887 EXPECT_EQ(oldOTSCount - 1, newOTSCount);
1888 EXPECT_EQ(oldOtherCount + 1, newOtherCount);
1891 TEST_F(BookmarkBarControllerDragDropTest, DragBookmarkData) {
1892 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1893 const BookmarkNode* root = model->bookmark_bar_node();
1894 const std::string model_string("1b 2f:[ 2f1b 2f2f:[ 2f2f1b 2f2f2b 2f2f3b ] "
1896 test::AddNodesFromModelString(model, root, model_string);
1897 const BookmarkNode* other = model->other_node();
1898 const std::string other_string("O1b O2b O3f:[ O3f1b O3f2f ] "
1899 "O4f:[ O4f1b O4f2f ] 05b ");
1900 test::AddNodesFromModelString(model, other, other_string);
1902 // Validate initial model.
1903 std::string actual = test::ModelStringFromNode(root);
1904 EXPECT_EQ(model_string, actual);
1905 actual = test::ModelStringFromNode(other);
1906 EXPECT_EQ(other_string, actual);
1908 // Remember the little ones.
1909 int oldChildCount = root->child_count();
1911 BookmarkButton* targetButton = [bar_ buttonWithTitleEqualTo:@"3b"];
1912 ASSERT_TRUE(targetButton);
1914 // Gen up some dragging data.
1915 const BookmarkNode* newNode = other->GetChild(2);
1916 [bar_ setDragDataNode:newNode];
1917 base::scoped_nsobject<FakeDragInfo> dragInfo([[FakeDragInfo alloc] init]);
1918 [dragInfo setDropLocation:[targetButton center]];
1919 [bar_ dragBookmarkData:(id<NSDraggingInfo>)dragInfo.get()];
1921 // There should one more button in the bar.
1922 int newChildCount = root->child_count();
1923 EXPECT_EQ(oldChildCount + 1, newChildCount);
1924 // Verify the model.
1925 const std::string expected("1b 2f:[ 2f1b 2f2f:[ 2f2f1b 2f2f2b 2f2f3b ] "
1926 "2f3b ] O3f:[ O3f1b O3f2f ] 3b 4b ");
1927 actual = test::ModelStringFromNode(root);
1928 EXPECT_EQ(expected, actual);
1929 oldChildCount = newChildCount;
1931 // Now do it over a folder button.
1932 targetButton = [bar_ buttonWithTitleEqualTo:@"2f"];
1933 ASSERT_TRUE(targetButton);
1934 NSPoint targetPoint = [targetButton center];
1935 newNode = other->GetChild(2); // Should be O4f.
1936 EXPECT_EQ(newNode->GetTitle(), ASCIIToUTF16("O4f"));
1937 [bar_ setDragDataNode:newNode];
1938 [dragInfo setDropLocation:targetPoint];
1939 [bar_ dragBookmarkData:(id<NSDraggingInfo>)dragInfo.get()];
1941 newChildCount = root->child_count();
1942 EXPECT_EQ(oldChildCount, newChildCount);
1943 // Verify the model.
1944 const std::string expected1("1b 2f:[ 2f1b 2f2f:[ 2f2f1b 2f2f2b 2f2f3b ] "
1945 "2f3b O4f:[ O4f1b O4f2f ] ] O3f:[ O3f1b O3f2f ] "
1947 actual = test::ModelStringFromNode(root);
1948 EXPECT_EQ(expected1, actual);
1951 TEST_F(BookmarkBarControllerDragDropTest, AddURLs) {
1952 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1953 const BookmarkNode* root = model->bookmark_bar_node();
1954 const std::string model_string("1b 2f:[ 2f1b 2f2f:[ 2f2f1b 2f2f2b 2f2f3b ] "
1956 test::AddNodesFromModelString(model, root, model_string);
1958 // Validate initial model.
1959 std::string actual = test::ModelStringFromNode(root);
1960 EXPECT_EQ(model_string, actual);
1962 // Remember the children.
1963 int oldChildCount = root->child_count();
1965 BookmarkButton* targetButton = [bar_ buttonWithTitleEqualTo:@"3b"];
1966 ASSERT_TRUE(targetButton);
1968 NSArray* urls = [NSArray arrayWithObjects: @"http://www.a.com/",
1969 @"http://www.b.com/", nil];
1970 NSArray* titles = [NSArray arrayWithObjects: @"SiteA", @"SiteB", nil];
1971 [bar_ addURLs:urls withTitles:titles at:[targetButton center]];
1973 // There should two more nodes in the bar.
1974 int newChildCount = root->child_count();
1975 EXPECT_EQ(oldChildCount + 2, newChildCount);
1976 // Verify the model.
1977 const std::string expected("1b 2f:[ 2f1b 2f2f:[ 2f2f1b 2f2f2b 2f2f3b ] "
1978 "2f3b ] SiteA SiteB 3b 4b ");
1979 actual = test::ModelStringFromNode(root);
1980 EXPECT_EQ(expected, actual);
1983 TEST_F(BookmarkBarControllerDragDropTest, ControllerForNode) {
1984 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
1985 const BookmarkNode* root = model->bookmark_bar_node();
1986 const std::string model_string("1b 2f:[ 2f1b 2f2b ] 3b ");
1987 test::AddNodesFromModelString(model, root, model_string);
1989 // Validate initial model.
1990 std::string actualModelString = test::ModelStringFromNode(root);
1991 EXPECT_EQ(model_string, actualModelString);
1993 // Find the main bar controller.
1994 const void* expectedController = bar_;
1995 const void* actualController = [bar_ controllerForNode:root];
1996 EXPECT_EQ(expectedController, actualController);
1999 TEST_F(BookmarkBarControllerDragDropTest, DropPositionIndicator) {
2000 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
2001 const BookmarkNode* root = model->bookmark_bar_node();
2002 const std::string model_string("1b 2f:[ 2f1b 2f2b 2f3b ] 3b 4b ");
2003 test::AddNodesFromModelString(model, root, model_string);
2005 // Hide the apps shortcut.
2006 profile()->GetPrefs()->SetBoolean(prefs::kShowAppsShortcutInBookmarkBar,
2008 ASSERT_TRUE([bar_ appsPageShortcutButtonIsHidden]);
2010 // Validate initial model.
2011 std::string actualModel = test::ModelStringFromNode(root);
2012 EXPECT_EQ(model_string, actualModel);
2014 // Test a series of points starting at the right edge of the bar.
2015 BookmarkButton* targetButton = [bar_ buttonWithTitleEqualTo:@"1b"];
2016 ASSERT_TRUE(targetButton);
2017 NSPoint targetPoint = [targetButton left];
2018 CGFloat leftMarginIndicatorPosition = bookmarks::kBookmarkLeftMargin - 0.5 *
2019 bookmarks::kBookmarkHorizontalPadding;
2020 const CGFloat baseOffset = targetPoint.x;
2021 CGFloat expected = leftMarginIndicatorPosition;
2022 CGFloat actual = [bar_ indicatorPosForDragToPoint:targetPoint];
2023 EXPECT_CGFLOAT_EQ(expected, actual);
2024 targetButton = [bar_ buttonWithTitleEqualTo:@"2f"];
2025 actual = [bar_ indicatorPosForDragToPoint:[targetButton right]];
2026 targetButton = [bar_ buttonWithTitleEqualTo:@"3b"];
2027 expected = [targetButton left].x - baseOffset + leftMarginIndicatorPosition;
2028 EXPECT_CGFLOAT_EQ(expected, actual);
2029 targetButton = [bar_ buttonWithTitleEqualTo:@"4b"];
2030 targetPoint = [targetButton right];
2031 targetPoint.x += 100; // Somewhere off to the right.
2032 CGFloat xDelta = 0.5 * bookmarks::kBookmarkHorizontalPadding;
2033 expected = NSMaxX([targetButton frame]) + xDelta;
2034 actual = [bar_ indicatorPosForDragToPoint:targetPoint];
2035 EXPECT_CGFLOAT_EQ(expected, actual);
2038 TEST_F(BookmarkBarControllerDragDropTest, PulseButton) {
2039 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
2040 const BookmarkNode* root = model->bookmark_bar_node();
2041 GURL gurl("http://www.google.com");
2042 const BookmarkNode* node = model->AddURL(root, root->child_count(),
2043 ASCIIToUTF16("title"), gurl);
2045 BookmarkButton* button = [[bar_ buttons] objectAtIndex:0];
2046 EXPECT_FALSE([button isContinuousPulsing]);
2048 NSValue *value = [NSValue valueWithPointer:node];
2049 NSDictionary *dict = [NSDictionary
2050 dictionaryWithObjectsAndKeys:value,
2051 bookmark_button::kBookmarkKey,
2052 [NSNumber numberWithBool:YES],
2053 bookmark_button::kBookmarkPulseFlagKey,
2055 [[NSNotificationCenter defaultCenter]
2056 postNotificationName:bookmark_button::kPulseBookmarkButtonNotification
2059 EXPECT_TRUE([button isContinuousPulsing]);
2061 dict = [NSDictionary dictionaryWithObjectsAndKeys:value,
2062 bookmark_button::kBookmarkKey,
2063 [NSNumber numberWithBool:NO],
2064 bookmark_button::kBookmarkPulseFlagKey,
2066 [[NSNotificationCenter defaultCenter]
2067 postNotificationName:bookmark_button::kPulseBookmarkButtonNotification
2070 EXPECT_FALSE([button isContinuousPulsing]);
2073 TEST_F(BookmarkBarControllerDragDropTest, DragBookmarkDataToTrash) {
2074 BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
2075 const BookmarkNode* root = model->bookmark_bar_node();
2076 const std::string model_string("1b 2f:[ 2f1b 2f2f:[ 2f2f1b 2f2f2b 2f2f3b ] "
2078 test::AddNodesFromModelString(model, root, model_string);
2080 // Validate initial model.
2081 std::string actual = test::ModelStringFromNode(root);
2082 EXPECT_EQ(model_string, actual);
2084 int oldChildCount = root->child_count();
2086 // Drag a button to the trash.
2087 BookmarkButton* buttonToDelete = [bar_ buttonWithTitleEqualTo:@"3b"];
2088 ASSERT_TRUE(buttonToDelete);
2089 EXPECT_TRUE([bar_ canDragBookmarkButtonToTrash:buttonToDelete]);
2090 [bar_ didDragBookmarkToTrash:buttonToDelete];
2092 // There should be one less button in the bar.
2093 int newChildCount = root->child_count();
2094 EXPECT_EQ(oldChildCount - 1, newChildCount);
2095 // Verify the model.
2096 const std::string expected("1b 2f:[ 2f1b 2f2f:[ 2f2f1b 2f2f2b 2f2f3b ] "
2098 actual = test::ModelStringFromNode(root);
2099 EXPECT_EQ(expected, actual);
2101 // Verify that the other bookmark folder can't be deleted.
2102 BookmarkButton *otherButton = [bar_ otherBookmarksButton];
2103 EXPECT_FALSE([bar_ canDragBookmarkButtonToTrash:otherButton]);