| OLD | NEW |
| 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #import "browser_actions_controller.h" | 5 #import "browser_actions_controller.h" |
| 6 | 6 |
| 7 #include <cmath> |
| 7 #include <string> | 8 #include <string> |
| 8 | 9 |
| 9 #include "app/resource_bundle.h" | 10 #include "app/resource_bundle.h" |
| 10 #include "base/sys_string_conversions.h" | 11 #include "base/sys_string_conversions.h" |
| 11 #include "chrome/browser/browser.h" | 12 #include "chrome/browser/browser.h" |
| 12 #include "chrome/browser/pref_service.h" | 13 #include "chrome/browser/pref_service.h" |
| 13 #import "chrome/browser/cocoa/extensions/browser_action_button.h" | 14 #import "chrome/browser/cocoa/extensions/browser_action_button.h" |
| 14 #import "chrome/browser/cocoa/extensions/browser_actions_container_view.h" | 15 #import "chrome/browser/cocoa/extensions/browser_actions_container_view.h" |
| 15 #import "chrome/browser/cocoa/extensions/extension_popup_controller.h" | 16 #import "chrome/browser/cocoa/extensions/extension_popup_controller.h" |
| 16 #import "chrome/browser/cocoa/menu_button.h" | 17 #import "chrome/browser/cocoa/menu_button.h" |
| (...skipping 35 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 52 // object since it is called when the toolbar model changes. | 53 // object since it is called when the toolbar model changes. |
| 53 - (void)createActionButtonForExtension:(Extension*)extension | 54 - (void)createActionButtonForExtension:(Extension*)extension |
| 54 withIndex:(NSUInteger)index; | 55 withIndex:(NSUInteger)index; |
| 55 // Removes an action button for the given extension from the container. This | 56 // Removes an action button for the given extension from the container. This |
| 56 // method also does not affect the underlying toolbar model since it is called | 57 // method also does not affect the underlying toolbar model since it is called |
| 57 // when the toolbar model changes. | 58 // when the toolbar model changes. |
| 58 - (void)removeActionButtonForExtension:(Extension*)extension; | 59 - (void)removeActionButtonForExtension:(Extension*)extension; |
| 59 // Useful in the case of a Browser Action being added/removed from the middle of | 60 // Useful in the case of a Browser Action being added/removed from the middle of |
| 60 // the container, this method repositions each button according to the current | 61 // the container, this method repositions each button according to the current |
| 61 // toolbar model. | 62 // toolbar model. |
| 62 - (void)repositionActionButtons; | 63 - (void)repositionActionButtonsAndAnimate:(BOOL)animate; |
| 63 // During container resizing, buttons become more transparent as they are pushed | 64 // During container resizing, buttons become more transparent as they are pushed |
| 64 // off the screen. This method updates each button's opacity determined by the | 65 // off the screen. This method updates each button's opacity determined by the |
| 65 // position of the button. | 66 // position of the button. |
| 66 - (void)updateButtonOpacity; | 67 - (void)updateButtonOpacity; |
| 67 // Returns the existing button with the given extension backing it; nil if it | 68 // Returns the existing button with the given extension backing it; nil if it |
| 68 // cannot be found or the extension's ID is invalid. | 69 // cannot be found or the extension's ID is invalid. |
| 69 - (BrowserActionButton*)buttonForExtension:(Extension*)extension; | 70 - (BrowserActionButton*)buttonForExtension:(Extension*)extension; |
| 70 // Returns the preferred width of the container given the number of visible | 71 // Returns the preferred width of the container given the number of visible |
| 71 // buttons |buttonCount|. | 72 // buttons |buttonCount|. |
| 72 - (CGFloat)containerWidthWithButtonCount:(NSUInteger)buttonCount; | 73 - (CGFloat)containerWidthWithButtonCount:(NSUInteger)buttonCount; |
| (...skipping 10 matching lines...) Expand all Loading... |
| 83 - (void)containerDragStart:(NSNotification*)notification; | 84 - (void)containerDragStart:(NSNotification*)notification; |
| 84 // Sends a notification for the toolbar to reposition surrounding UI elements. | 85 // Sends a notification for the toolbar to reposition surrounding UI elements. |
| 85 - (void)containerDragging:(NSNotification*)notification; | 86 - (void)containerDragging:(NSNotification*)notification; |
| 86 // Determines which buttons need to be hidden based on the new size, hides them | 87 // Determines which buttons need to be hidden based on the new size, hides them |
| 87 // and updates the chevron overflow menu. Also fires a notification to let the | 88 // and updates the chevron overflow menu. Also fires a notification to let the |
| 88 // toolbar know that the drag has finished. | 89 // toolbar know that the drag has finished. |
| 89 - (void)containerDragFinished:(NSNotification*)notification; | 90 - (void)containerDragFinished:(NSNotification*)notification; |
| 90 // Updates the image associated with the button should it be within the chevron | 91 // Updates the image associated with the button should it be within the chevron |
| 91 // menu. | 92 // menu. |
| 92 - (void)actionButtonUpdated:(NSNotification*)notification; | 93 - (void)actionButtonUpdated:(NSNotification*)notification; |
| 93 | 94 // Adjusts the position of the surrounding action buttons depending on where the |
| 95 // button is within the container. |
| 96 - (void)actionButtonDragging:(NSNotification*)notification; |
| 97 // Updates the underlying toolbar model and "snaps" the button into its proper |
| 98 // place within the button grid. |
| 99 - (void)actionButtonDragFinished:(NSNotification*)notification; |
| 100 // Moves the given button both visually and within the toolbar model to the |
| 101 // specified index. |
| 102 - (void)moveButton:(BrowserActionButton*)button |
| 103 toIndex:(NSUInteger)index |
| 104 animate:(BOOL)animate; |
| 94 // Handles when the given BrowserActionButton object is clicked. | 105 // Handles when the given BrowserActionButton object is clicked. |
| 95 - (void)browserActionClicked:(BrowserActionButton*)button; | 106 - (void)browserActionClicked:(BrowserActionButton*)button; |
| 96 // Returns whether the given extension should be displayed. Only displays | 107 // Returns whether the given extension should be displayed. Only displays |
| 97 // incognito-enabled extensions in incognito mode. Otherwise returns YES. | 108 // incognito-enabled extensions in incognito mode. Otherwise returns YES. |
| 98 - (BOOL)shouldDisplayBrowserAction:(Extension*)extension; | 109 - (BOOL)shouldDisplayBrowserAction:(Extension*)extension; |
| 99 | 110 |
| 100 // The reason |frame| is specified in these chevron functions is because the | 111 // The reason |frame| is specified in these chevron functions is because the |
| 101 // container may be animating and the end frame of the animation should be | 112 // container may be animating and the end frame of the animation should be |
| 102 // passed instead of the current frame (which may be off and cause the chevron | 113 // passed instead of the current frame (which may be off and cause the chevron |
| 103 // to jump at the end of its animation). | 114 // to jump at the end of its animation). |
| (...skipping 313 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 417 | 428 |
| 418 [buttons_ setObject:newButton forKey:buttonKey]; | 429 [buttons_ setObject:newButton forKey:buttonKey]; |
| 419 if (index < [self containerButtonCapacity]) { | 430 if (index < [self containerButtonCapacity]) { |
| 420 [containerView_ addSubview:newButton]; | 431 [containerView_ addSubview:newButton]; |
| 421 } else { | 432 } else { |
| 422 [hiddenButtons_ addObject:newButton]; | 433 [hiddenButtons_ addObject:newButton]; |
| 423 [newButton setAlphaValue:0.0]; | 434 [newButton setAlphaValue:0.0]; |
| 424 [self updateOverflowMenu]; | 435 [self updateOverflowMenu]; |
| 425 } | 436 } |
| 426 | 437 |
| 427 [self repositionActionButtons]; | 438 [[NSNotificationCenter defaultCenter] |
| 439 addObserver:self |
| 440 selector:@selector(actionButtonDragging:) |
| 441 name:kBrowserActionButtonDraggingNotification |
| 442 object:newButton]; |
| 443 [[NSNotificationCenter defaultCenter] |
| 444 addObserver:self |
| 445 selector:@selector(actionButtonDragFinished:) |
| 446 name:kBrowserActionButtonDragEndNotification |
| 447 object:newButton]; |
| 448 |
| 449 [self repositionActionButtonsAndAnimate:NO]; |
| 428 [containerView_ setMaxWidth: | 450 [containerView_ setMaxWidth: |
| 429 [self containerWidthWithButtonCount:[self buttonCount]]]; | 451 [self containerWidthWithButtonCount:[self buttonCount]]]; |
| 430 [containerView_ setNeedsDisplay:YES]; | 452 [containerView_ setNeedsDisplay:YES]; |
| 431 } | 453 } |
| 432 | 454 |
| 433 - (void)removeActionButtonForExtension:(Extension*)extension { | 455 - (void)removeActionButtonForExtension:(Extension*)extension { |
| 434 if (!extension->browser_action()) | 456 if (!extension->browser_action()) |
| 435 return; | 457 return; |
| 436 | 458 |
| 437 NSString* buttonKey = base::SysUTF8ToNSString(extension->id()); | 459 NSString* buttonKey = base::SysUTF8ToNSString(extension->id()); |
| (...skipping 10 matching lines...) Expand all Loading... |
| 448 // It may or may not be hidden, but it won't matter to NSMutableArray either | 470 // It may or may not be hidden, but it won't matter to NSMutableArray either |
| 449 // way. | 471 // way. |
| 450 [hiddenButtons_ removeObject:button]; | 472 [hiddenButtons_ removeObject:button]; |
| 451 [self updateOverflowMenu]; | 473 [self updateOverflowMenu]; |
| 452 | 474 |
| 453 [buttons_ removeObjectForKey:buttonKey]; | 475 [buttons_ removeObjectForKey:buttonKey]; |
| 454 if ([self buttonCount] == 0) { | 476 if ([self buttonCount] == 0) { |
| 455 // No more buttons? Hide the container. | 477 // No more buttons? Hide the container. |
| 456 [containerView_ setHidden:YES]; | 478 [containerView_ setHidden:YES]; |
| 457 } else { | 479 } else { |
| 458 [self repositionActionButtons]; | 480 [self repositionActionButtonsAndAnimate:NO]; |
| 459 } | 481 } |
| 460 [containerView_ setMaxWidth: | 482 [containerView_ setMaxWidth: |
| 461 [self containerWidthWithButtonCount:[self buttonCount]]]; | 483 [self containerWidthWithButtonCount:[self buttonCount]]]; |
| 462 [containerView_ setNeedsDisplay:YES]; | 484 [containerView_ setNeedsDisplay:YES]; |
| 463 } | 485 } |
| 464 | 486 |
| 465 - (void)repositionActionButtons { | 487 - (void)repositionActionButtonsAndAnimate:(BOOL)animate { |
| 466 NSUInteger i = 0; | 488 NSUInteger i = 0; |
| 467 for (ExtensionList::iterator iter = toolbarModel_->begin(); | 489 for (ExtensionList::iterator iter = toolbarModel_->begin(); |
| 468 iter != toolbarModel_->end(); ++iter) { | 490 iter != toolbarModel_->end(); ++iter) { |
| 469 if (![self shouldDisplayBrowserAction:*iter]) | 491 if (![self shouldDisplayBrowserAction:*iter]) |
| 470 continue; | 492 continue; |
| 471 | |
| 472 CGFloat xOffset = kGrippyXOffset + | |
| 473 (i * (kBrowserActionWidth + kBrowserActionButtonPadding)); | |
| 474 BrowserActionButton* button = [self buttonForExtension:(*iter)]; | 493 BrowserActionButton* button = [self buttonForExtension:(*iter)]; |
| 475 if (!button) | 494 if (!button) |
| 476 continue; | 495 continue; |
| 477 NSRect buttonFrame = [button frame]; | 496 if (![button isBeingDragged]) |
| 478 buttonFrame.origin.x = xOffset; | 497 [self moveButton:button toIndex:i animate:animate]; |
| 479 [button setFrame:buttonFrame]; | |
| 480 ++i; | 498 ++i; |
| 481 } | 499 } |
| 482 } | 500 } |
| 483 | 501 |
| 484 - (void)updateButtonOpacity { | 502 - (void)updateButtonOpacity { |
| 485 for (BrowserActionButton* button in [buttons_ allValues]) { | 503 for (BrowserActionButton* button in [buttons_ allValues]) { |
| 486 NSRect buttonFrame = [button frame]; | 504 NSRect buttonFrame = [button frame]; |
| 487 buttonFrame.origin.x += kButtonOpacityLeadPadding; | 505 buttonFrame.origin.x += kButtonOpacityLeadPadding; |
| 488 if (NSContainsRect([containerView_ bounds], buttonFrame)) { | 506 if (NSContainsRect([containerView_ bounds], buttonFrame)) { |
| 489 if ([button alphaValue] != 1.0) | 507 if ([button alphaValue] != 1.0) |
| (...skipping 90 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 580 if (![hiddenButtons_ containsObject:button]) | 598 if (![hiddenButtons_ containsObject:button]) |
| 581 return; | 599 return; |
| 582 | 600 |
| 583 // +1 item because of the title placeholder. See |updateOverflowMenu|. | 601 // +1 item because of the title placeholder. See |updateOverflowMenu|. |
| 584 NSUInteger menuIndex = [hiddenButtons_ indexOfObject:button] + 1; | 602 NSUInteger menuIndex = [hiddenButtons_ indexOfObject:button] + 1; |
| 585 NSMenuItem* item = [[chevronMenuButton_ attachedMenu] itemAtIndex:menuIndex]; | 603 NSMenuItem* item = [[chevronMenuButton_ attachedMenu] itemAtIndex:menuIndex]; |
| 586 DCHECK(button == [item representedObject]); | 604 DCHECK(button == [item representedObject]); |
| 587 [item setImage:[button compositedImage]]; | 605 [item setImage:[button compositedImage]]; |
| 588 } | 606 } |
| 589 | 607 |
| 608 - (void)actionButtonDragging:(NSNotification*)notification { |
| 609 if (![self chevronIsHidden]) |
| 610 [self setChevronHidden:YES inFrame:[containerView_ frame] animate:YES]; |
| 611 |
| 612 // Determine what index the dragged button should lie in, alter the model and |
| 613 // reposition the buttons. |
| 614 CGFloat dragThreshold = std::floor(kBrowserActionWidth / 2); |
| 615 BrowserActionButton* draggedButton = [notification object]; |
| 616 NSRect draggedButtonFrame = [draggedButton frame]; |
| 617 |
| 618 NSUInteger index = 0; |
| 619 for (ExtensionList::iterator iter = toolbarModel_->begin(); |
| 620 iter != toolbarModel_->end(); ++iter) { |
| 621 BrowserActionButton* button = [self buttonForExtension:(*iter)]; |
| 622 CGFloat intersectionWidth = |
| 623 NSWidth(NSIntersectionRect(draggedButtonFrame, [button frame])); |
| 624 |
| 625 if (intersectionWidth > dragThreshold && button != draggedButton && |
| 626 ![button isAnimating]) { |
| 627 toolbarModel_->MoveBrowserAction([draggedButton extension], index); |
| 628 [self repositionActionButtonsAndAnimate:YES]; |
| 629 return; |
| 630 } |
| 631 ++index; |
| 632 } |
| 633 } |
| 634 |
| 635 - (void)actionButtonDragFinished:(NSNotification*)notification { |
| 636 [self showChevronIfNecessaryInFrame:[containerView_ frame] animate:YES]; |
| 637 DCHECK(![[notification object] isBeingDragged]); |
| 638 [self repositionActionButtonsAndAnimate:YES]; |
| 639 } |
| 640 |
| 641 - (void)moveButton:(BrowserActionButton*)button |
| 642 toIndex:(NSUInteger)index |
| 643 animate:(BOOL)animate { |
| 644 CGFloat xOffset = kGrippyXOffset + |
| 645 (index * (kBrowserActionWidth + kBrowserActionButtonPadding)); |
| 646 NSRect buttonFrame = [button frame]; |
| 647 buttonFrame.origin.x = xOffset; |
| 648 [button setFrame:buttonFrame animate:animate]; |
| 649 } |
| 650 |
| 590 - (void)browserActionClicked:(BrowserActionButton*)button { | 651 - (void)browserActionClicked:(BrowserActionButton*)button { |
| 591 int tabId = [self currentTabId]; | 652 int tabId = [self currentTabId]; |
| 592 if (tabId < 0) { | 653 if (tabId < 0) { |
| 593 NOTREACHED() << "No current tab."; | 654 NOTREACHED() << "No current tab."; |
| 594 return; | 655 return; |
| 595 } | 656 } |
| 596 | 657 |
| 597 ExtensionAction* action = [button extension]->browser_action(); | 658 ExtensionAction* action = [button extension]->browser_action(); |
| 598 if (action->HasPopup(tabId)) { | 659 if (action->HasPopup(tabId)) { |
| 599 NSPoint arrowPoint = [self popupPointForBrowserAction:[button extension]]; | 660 NSPoint arrowPoint = [self popupPointForBrowserAction:[button extension]]; |
| (...skipping 62 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 662 NSViewAnimationFadeOutEffect, NSViewAnimationEffectKey, | 723 NSViewAnimationFadeOutEffect, NSViewAnimationEffectKey, |
| 663 nil]; | 724 nil]; |
| 664 } else { | 725 } else { |
| 665 [chevronMenuButton_ setHidden:NO]; | 726 [chevronMenuButton_ setHidden:NO]; |
| 666 animationDictionary = [NSDictionary dictionaryWithObjectsAndKeys: | 727 animationDictionary = [NSDictionary dictionaryWithObjectsAndKeys: |
| 667 chevronMenuButton_.get(), NSViewAnimationTargetKey, | 728 chevronMenuButton_.get(), NSViewAnimationTargetKey, |
| 668 NSViewAnimationFadeInEffect, NSViewAnimationEffectKey, | 729 NSViewAnimationFadeInEffect, NSViewAnimationEffectKey, |
| 669 nil]; | 730 nil]; |
| 670 } | 731 } |
| 671 [chevronAnimation_ setViewAnimations: | 732 [chevronAnimation_ setViewAnimations: |
| 672 [NSArray arrayWithObjects:animationDictionary, nil]]; | 733 [NSArray arrayWithObject:animationDictionary]]; |
| 673 [chevronAnimation_ startAnimation]; | 734 [chevronAnimation_ startAnimation]; |
| 674 } | 735 } |
| 675 | 736 |
| 676 - (void)chevronItemSelected:(id)menuItem { | 737 - (void)chevronItemSelected:(id)menuItem { |
| 677 [self browserActionClicked:[menuItem representedObject]]; | 738 [self browserActionClicked:[menuItem representedObject]]; |
| 678 } | 739 } |
| 679 | 740 |
| 680 - (void)updateOverflowMenu { | 741 - (void)updateOverflowMenu { |
| 681 overflowMenu_.reset([[NSMenu alloc] initWithTitle:@""]); | 742 overflowMenu_.reset([[NSMenu alloc] initWithTitle:@""]); |
| 682 // See menu_button.h for documentation on why this is needed. | 743 // See menu_button.h for documentation on why this is needed. |
| (...skipping 29 matching lines...) Expand all Loading... |
| 712 iter != toolbarModel_->end(); ++iter) { | 773 iter != toolbarModel_->end(); ++iter) { |
| 713 if (i == index) | 774 if (i == index) |
| 714 return [buttons_ objectForKey:base::SysUTF8ToNSString((*iter)->id())]; | 775 return [buttons_ objectForKey:base::SysUTF8ToNSString((*iter)->id())]; |
| 715 | 776 |
| 716 ++i; | 777 ++i; |
| 717 } | 778 } |
| 718 return nil; | 779 return nil; |
| 719 } | 780 } |
| 720 | 781 |
| 721 @end | 782 @end |
| OLD | NEW |