Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 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 "chrome/browser/ui/cocoa/bookmarks/bookmark_button.h" | 5 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_button.h" |
| 6 | 6 |
| 7 #include <cmath> | 7 #include <cmath> |
| 8 | 8 |
| 9 #include "base/logging.h" | 9 #include "base/logging.h" |
| 10 #include "base/mac/foundation_util.h" | 10 #include "base/mac/foundation_util.h" |
| 11 #import "base/mac/scoped_nsobject.h" | 11 #import "base/mac/scoped_nsobject.h" |
| 12 #include "base/mac/sdk_forward_declarations.h" | |
| 12 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_bar_folder_window.h" | 13 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_bar_folder_window.h" |
| 13 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_button_cell.h" | 14 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_button_cell.h" |
| 15 #import "chrome/browser/ui/cocoa/bookmarks/bookmark_folder_target.h" | |
| 14 #import "chrome/browser/ui/cocoa/browser_window_controller.h" | 16 #import "chrome/browser/ui/cocoa/browser_window_controller.h" |
| 15 #import "chrome/browser/ui/cocoa/view_id_util.h" | 17 #import "chrome/browser/ui/cocoa/view_id_util.h" |
| 16 #include "components/bookmarks/browser/bookmark_model.h" | 18 #include "components/bookmarks/browser/bookmark_model.h" |
| 17 #include "content/public/browser/user_metrics.h" | 19 #include "content/public/browser/user_metrics.h" |
| 18 #include "ui/base/cocoa/cocoa_base_utils.h" | 20 #include "ui/base/cocoa/cocoa_base_utils.h" |
| 19 #import "ui/base/cocoa/nsview_additions.h" | 21 #import "ui/base/cocoa/nsview_additions.h" |
| 20 #include "ui/gfx/scoped_ns_graphics_context_save_gstate_mac.h" | 22 #include "ui/gfx/scoped_ns_graphics_context_save_gstate_mac.h" |
| 21 | 23 |
| 22 using base::UserMetricsAction; | 24 using base::UserMetricsAction; |
| 23 using bookmarks::BookmarkNode; | 25 using bookmarks::BookmarkNode; |
| 24 | 26 |
| 25 // The opacity of the bookmark button drag image. | 27 // The opacity of the bookmark button drag image. |
| 26 static const CGFloat kDragImageOpacity = 0.7; | 28 static const CGFloat kDragImageOpacity = 0.7; |
| 27 | 29 |
| 28 namespace { | 30 namespace { |
| 29 // We need a class variable to track the current dragged button to enable | 31 // We need a class variable to track the current dragged button to enable |
| 30 // proper live animated dragging behavior, and can't do it in the | 32 // proper live animated dragging behavior, and can't do it in the |
| 31 // delegate/controller since you can drag a button from one domain to the | 33 // delegate/controller since you can drag a button from one domain to the |
| 32 // other (from a "folder" menu, to the main bar, or vice versa). | 34 // other (from a "folder" menu, to the main bar, or vice versa). |
| 33 BookmarkButton* gDraggedButton = nil; // Weak | 35 BookmarkButton* gDraggedButton = nil; // Weak |
| 34 }; | 36 }; |
| 35 | 37 |
| 36 @interface BookmarkButton(Private) | 38 @interface BookmarkButton() <NSPasteboardItemDataProvider> |
| 39 | |
| 40 // NSPasteboardItemDataProvider: | |
| 41 - (void)pasteboard:(NSPasteboard*)sender | |
| 42 item:(NSPasteboardItem*)item | |
| 43 provideDataForType:(NSString*)type; | |
| 44 | |
| 45 // NSDraggingSource: | |
| 46 - (void)draggingSession:(NSDraggingSession*)session | |
| 47 endedAtPoint:(NSPoint)aPoint | |
| 48 operation:(NSDragOperation)operation; | |
| 49 - (NSDragOperation)draggingSession:(NSDraggingSession*)session | |
| 50 sourceOperationMaskForDraggingContext:(NSDraggingContext)context; | |
| 37 | 51 |
| 38 // Make a drag image for the button. | 52 // Make a drag image for the button. |
| 39 - (NSImage*)dragImage; | 53 - (NSImage*)dragImage; |
| 40 | 54 |
| 41 - (void)installCustomTrackingArea; | 55 - (void)installCustomTrackingArea; |
| 42 | 56 |
| 43 @end // @interface BookmarkButton(Private) | 57 @end // @interface BookmarkButton(Private) |
| 44 | 58 |
| 45 | |
| 46 @implementation BookmarkButton | 59 @implementation BookmarkButton |
| 47 | 60 |
| 48 @synthesize delegate = delegate_; | 61 @synthesize delegate = delegate_; |
| 49 @synthesize acceptsTrackIn = acceptsTrackIn_; | 62 @synthesize acceptsTrackIn = acceptsTrackIn_; |
| 50 | 63 |
| 51 - (id)initWithFrame:(NSRect)frameRect { | 64 - (id)initWithFrame:(NSRect)frameRect { |
| 52 // BookmarkButton's ViewID may be changed to VIEW_ID_OTHER_BOOKMARKS in | 65 // BookmarkButton's ViewID may be changed to VIEW_ID_OTHER_BOOKMARKS in |
| 53 // BookmarkBarController, so we can't just override -viewID method to return | 66 // BookmarkBarController, so we can't just override -viewID method to return |
| 54 // it. | 67 // it. |
| 55 if ((self = [super initWithFrame:frameRect])) { | 68 if ((self = [super initWithFrame:frameRect])) { |
| (...skipping 112 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 168 if ([self isFolder]) { | 181 if ([self isFolder]) { |
| 169 // Close the folder's drop-down menu if it's visible. | 182 // Close the folder's drop-down menu if it's visible. |
| 170 [[self target] closeBookmarkFolder:self]; | 183 [[self target] closeBookmarkFolder:self]; |
| 171 } | 184 } |
| 172 | 185 |
| 173 // At the moment, moving bookmarks causes their buttons (like me!) | 186 // At the moment, moving bookmarks causes their buttons (like me!) |
| 174 // to be destroyed and rebuilt. Make sure we don't go away while on | 187 // to be destroyed and rebuilt. Make sure we don't go away while on |
| 175 // the stack. | 188 // the stack. |
| 176 [self retain]; | 189 [self retain]; |
| 177 | 190 |
| 178 // Ask our delegate to fill the pasteboard for us. | |
| 179 NSPasteboard* pboard = [NSPasteboard pasteboardWithName:NSDragPboard]; | |
| 180 [[self delegate] fillPasteboard:pboard forDragOfButton:self]; | |
| 181 | |
| 182 // Lock bar visibility, forcing the overlay to stay visible if we are in | 191 // Lock bar visibility, forcing the overlay to stay visible if we are in |
| 183 // fullscreen mode. | 192 // fullscreen mode. |
| 184 if ([[self delegate] dragShouldLockBarVisibility]) { | 193 if ([[self delegate] dragShouldLockBarVisibility]) { |
| 185 DCHECK(!visibilityDelegate_); | 194 DCHECK(!visibilityDelegate_); |
| 186 NSWindow* window = [[self delegate] browserWindow]; | 195 NSWindow* window = [[self delegate] browserWindow]; |
| 187 visibilityDelegate_ = | 196 visibilityDelegate_ = |
| 188 [BrowserWindowController browserWindowControllerForWindow:window]; | 197 [BrowserWindowController browserWindowControllerForWindow:window]; |
| 189 [visibilityDelegate_ lockBarVisibilityForOwner:self | 198 [visibilityDelegate_ lockBarVisibilityForOwner:self |
| 190 withAnimation:NO | 199 withAnimation:NO |
| 191 delay:NO]; | 200 delay:NO]; |
| 192 } | 201 } |
| 193 const BookmarkNode* node = [self bookmarkNode]; | 202 const BookmarkNode* node = [self bookmarkNode]; |
| 194 const BookmarkNode* parent = node ? node->parent() : NULL; | 203 const BookmarkNode* parent = node ? node->parent() : NULL; |
| 195 if (parent && parent->type() == BookmarkNode::FOLDER) { | 204 if (parent && parent->type() == BookmarkNode::FOLDER) { |
| 196 content::RecordAction(UserMetricsAction("BookmarkBarFolder_DragStart")); | 205 content::RecordAction(UserMetricsAction("BookmarkBarFolder_DragStart")); |
| 197 } else { | 206 } else { |
| 198 content::RecordAction(UserMetricsAction("BookmarkBar_DragStart")); | 207 content::RecordAction(UserMetricsAction("BookmarkBar_DragStart")); |
| 199 } | 208 } |
| 200 | 209 |
| 201 dragMouseOffset_ = [self convertPoint:[event locationInWindow] fromView:nil]; | 210 dragMouseOffset_ = [self convertPoint:[event locationInWindow] fromView:nil]; |
| 202 dragPending_ = YES; | 211 dragPending_ = YES; |
| 203 gDraggedButton = self; | 212 gDraggedButton = self; |
| 204 | 213 |
| 205 CGFloat yAt = [self bounds].size.height; | |
| 206 NSSize dragOffset = NSMakeSize(0.0, 0.0); | |
| 207 NSImage* image = [self dragImage]; | 214 NSImage* image = [self dragImage]; |
| 208 [self setHidden:YES]; | 215 [self setHidden:YES]; |
| 209 [self dragImage:image at:NSMakePoint(0, yAt) offset:dragOffset | 216 |
| 210 event:event pasteboard:pboard source:self slideBack:YES]; | 217 NSPasteboardItem *pbItem = [NSPasteboardItem new]; |
| 218 [pbItem | |
| 219 setDataProvider:self | |
| 220 forTypes:[NSArray arrayWithObjects:kBookmarkButtonDragType, nil]]; | |
|
Avi (use Gerrit)
2016/03/18 21:41:03
@[ kBookmarkButtonDragType ]
?
erikchen
2016/03/18 22:20:35
Done.
| |
| 221 | |
| 222 base::scoped_nsobject<NSDraggingItem> dragItem( | |
| 223 [[NSDraggingItem alloc] initWithPasteboardWriter:pbItem]); | |
| 224 [dragItem setDraggingFrame:[self bounds] contents:image]; | |
| 225 | |
| 226 [self beginDraggingSessionWithItems:@[ dragItem.get() ] | |
| 227 event:event | |
| 228 source:self]; | |
| 229 while (gDraggedButton != nil) { | |
| 230 [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode | |
| 231 beforeDate:[NSDate distantFuture]]; | |
| 232 } | |
| 211 [self setHidden:NO]; | 233 [self setHidden:NO]; |
| 212 | 234 |
| 213 // And we're done. | 235 // And we're done. |
| 214 dragPending_ = NO; | 236 dragPending_ = NO; |
| 215 gDraggedButton = nil; | 237 gDraggedButton = nil; |
| 216 | 238 |
| 217 [self autorelease]; | 239 [self autorelease]; |
| 218 } | 240 } |
| 219 | 241 |
| 242 - (void)pasteboard:(NSPasteboard*)sender | |
| 243 item:(NSPasteboardItem*)item | |
| 244 provideDataForType:(NSString*)type { | |
| 245 [sender setData:[NSData dataWithBytes:&gDraggedButton | |
| 246 length:sizeof(gDraggedButton)] | |
| 247 forType:kBookmarkButtonDragType]; | |
| 248 } | |
| 249 | |
| 250 - (NSDragOperation)draggingSession:(NSDraggingSession*)session | |
| 251 sourceOperationMaskForDraggingContext:(NSDraggingContext)context { | |
| 252 NSDragOperation operation = NSDragOperationCopy; | |
| 253 | |
| 254 if (context == NSDraggingContextWithinApplication) | |
| 255 operation |= NSDragOperationMove; | |
| 256 | |
| 257 if ([delegate_ canDragBookmarkButtonToTrash:self]) | |
| 258 operation |= NSDragOperationDelete; | |
| 259 | |
| 260 return operation; | |
| 261 } | |
| 262 | |
| 220 // Overridden to release bar visibility. | 263 // Overridden to release bar visibility. |
| 221 - (DraggableButtonResult)endDrag { | 264 - (DraggableButtonResult)endDrag { |
| 222 gDraggedButton = nil; | 265 gDraggedButton = nil; |
| 223 | 266 |
| 224 // visibilityDelegate_ can be nil if we're detached, and that's fine. | 267 // visibilityDelegate_ can be nil if we're detached, and that's fine. |
| 225 [visibilityDelegate_ releaseBarVisibilityForOwner:self | 268 [visibilityDelegate_ releaseBarVisibilityForOwner:self |
| 226 withAnimation:YES | 269 withAnimation:YES |
| 227 delay:YES]; | 270 delay:YES]; |
| 228 visibilityDelegate_ = nil; | 271 visibilityDelegate_ = nil; |
| 229 | 272 |
| 230 return kDraggableButtonImplUseBase; | 273 return kDraggableButtonImplUseBase; |
| 231 } | 274 } |
| 232 | 275 |
| 233 - (NSDragOperation)draggingSourceOperationMaskForLocal:(BOOL)isLocal { | 276 - (void)draggingSession:(NSDraggingSession*)session |
| 234 NSDragOperation operation = NSDragOperationCopy; | 277 endedAtPoint:(NSPoint)aPoint |
| 235 if (isLocal) { | 278 operation:(NSDragOperation)operation { |
| 236 operation |= NSDragOperationMove; | |
| 237 } | |
| 238 if ([delegate_ canDragBookmarkButtonToTrash:self]) { | |
| 239 operation |= NSDragOperationDelete; | |
| 240 } | |
| 241 return operation; | |
| 242 } | |
| 243 | |
| 244 - (void)draggedImage:(NSImage *)anImage | |
| 245 endedAt:(NSPoint)aPoint | |
| 246 operation:(NSDragOperation)operation { | |
| 247 gDraggedButton = nil; | 279 gDraggedButton = nil; |
| 248 // Inform delegate of drag source that we're finished dragging, | 280 // Inform delegate of drag source that we're finished dragging, |
| 249 // so it can close auto-opened bookmark folders etc. | 281 // so it can close auto-opened bookmark folders etc. |
| 250 [delegate_ bookmarkDragDidEnd:self | 282 [delegate_ bookmarkDragDidEnd:self |
| 251 operation:operation]; | 283 operation:operation]; |
| 252 // Tell delegate if it should delete us. | 284 // Tell delegate if it should delete us. |
| 253 if (operation & NSDragOperationDelete) { | 285 if (operation & NSDragOperationDelete) { |
| 254 dragEndScreenLocation_ = aPoint; | 286 dragEndScreenLocation_ = aPoint; |
| 255 [delegate_ didDragBookmarkToTrash:self]; | 287 [delegate_ didDragBookmarkToTrash:self]; |
| 256 } | 288 } |
| (...skipping 61 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 318 break; | 350 break; |
| 319 } | 351 } |
| 320 default: | 352 default: |
| 321 /* Ignore any other kind of event. */ | 353 /* Ignore any other kind of event. */ |
| 322 break; | 354 break; |
| 323 } | 355 } |
| 324 } | 356 } |
| 325 return kDraggableButtonMixinDidWork; | 357 return kDraggableButtonMixinDidWork; |
| 326 } | 358 } |
| 327 | 359 |
| 328 | |
| 329 | |
| 330 // mouseEntered: and mouseExited: are called from our | 360 // mouseEntered: and mouseExited: are called from our |
| 331 // BookmarkButtonCell. We redirect this information to our delegate. | 361 // BookmarkButtonCell. We redirect this information to our delegate. |
| 332 // The controller can then perform menu-like actions (e.g. "hover over | 362 // The controller can then perform menu-like actions (e.g. "hover over |
| 333 // to open menu"). | 363 // to open menu"). |
| 334 - (void)mouseEntered:(NSEvent*)event { | 364 - (void)mouseEntered:(NSEvent*)event { |
| 335 [delegate_ mouseEnteredButton:self event:event]; | 365 [delegate_ mouseEnteredButton:self event:event]; |
| 336 } | 366 } |
| 337 | 367 |
| 338 // See comments above mouseEntered:. | 368 // See comments above mouseEntered:. |
| 339 - (void)mouseExited:(NSEvent*)event { | 369 - (void)mouseExited:(NSEvent*)event { |
| (...skipping 81 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 421 // ThemedWindowDrawing implementation. | 451 // ThemedWindowDrawing implementation. |
| 422 | 452 |
| 423 - (void)windowDidChangeTheme { | 453 - (void)windowDidChangeTheme { |
| 424 [self setNeedsDisplay:YES]; | 454 [self setNeedsDisplay:YES]; |
| 425 } | 455 } |
| 426 | 456 |
| 427 - (void)windowDidChangeActive { | 457 - (void)windowDidChangeActive { |
| 428 [self setNeedsDisplay:YES]; | 458 [self setNeedsDisplay:YES]; |
| 429 } | 459 } |
| 430 | 460 |
| 431 @end | |
| 432 | |
| 433 @implementation BookmarkButton(Private) | |
| 434 | |
| 435 | |
| 436 - (void)installCustomTrackingArea { | 461 - (void)installCustomTrackingArea { |
| 437 const NSTrackingAreaOptions options = | 462 const NSTrackingAreaOptions options = |
| 438 NSTrackingActiveAlways | | 463 NSTrackingActiveAlways | |
| 439 NSTrackingMouseEnteredAndExited | | 464 NSTrackingMouseEnteredAndExited | |
| 440 NSTrackingEnabledDuringMouseDrag; | 465 NSTrackingEnabledDuringMouseDrag; |
| 441 | 466 |
| 442 if (area_) { | 467 if (area_) { |
| 443 [self removeTrackingArea:area_]; | 468 [self removeTrackingArea:area_]; |
| 444 [area_ release]; | 469 [area_ release]; |
| 445 } | 470 } |
| 446 | 471 |
| 447 area_ = [[NSTrackingArea alloc] initWithRect:[self bounds] | 472 area_ = [[NSTrackingArea alloc] initWithRect:[self bounds] |
| 448 options:options | 473 options:options |
| 449 owner:self | 474 owner:self |
| 450 userInfo:nil]; | 475 userInfo:nil]; |
| 451 [self addTrackingArea:area_]; | 476 [self addTrackingArea:area_]; |
| 452 } | 477 } |
| 453 | 478 |
| 454 | |
| 455 - (NSImage*)dragImage { | 479 - (NSImage*)dragImage { |
| 456 NSRect bounds = [self bounds]; | 480 NSRect bounds = [self bounds]; |
| 457 base::scoped_nsobject<NSImage> image( | 481 base::scoped_nsobject<NSImage> image( |
| 458 [[NSImage alloc] initWithSize:bounds.size]); | 482 [[NSImage alloc] initWithSize:bounds.size]); |
| 459 [image lockFocusFlipped:[self isFlipped]]; | 483 [image lockFocusFlipped:[self isFlipped]]; |
| 460 | 484 |
| 461 NSGraphicsContext* context = [NSGraphicsContext currentContext]; | 485 NSGraphicsContext* context = [NSGraphicsContext currentContext]; |
| 462 CGContextRef cgContext = static_cast<CGContextRef>([context graphicsPort]); | 486 CGContextRef cgContext = static_cast<CGContextRef>([context graphicsPort]); |
| 463 CGContextBeginTransparencyLayer(cgContext, 0); | 487 CGContextBeginTransparencyLayer(cgContext, 0); |
| 464 CGContextSetAlpha(cgContext, kDragImageOpacity); | 488 CGContextSetAlpha(cgContext, kDragImageOpacity); |
| 465 | 489 |
| 466 GradientButtonCell* cell = | 490 GradientButtonCell* cell = |
| 467 base::mac::ObjCCastStrict<GradientButtonCell>([self cell]); | 491 base::mac::ObjCCastStrict<GradientButtonCell>([self cell]); |
| 468 [[cell clipPathForFrame:bounds inView:self] setClip]; | 492 [[cell clipPathForFrame:bounds inView:self] setClip]; |
| 469 [cell drawWithFrame:bounds inView:self]; | 493 [cell drawWithFrame:bounds inView:self]; |
| 470 | 494 |
| 471 CGContextEndTransparencyLayer(cgContext); | 495 CGContextEndTransparencyLayer(cgContext); |
| 472 [image unlockFocus]; | 496 [image unlockFocus]; |
| 473 | 497 |
| 474 return image.autorelease(); | 498 return image.autorelease(); |
| 475 } | 499 } |
| 476 | 500 |
| 477 @end // @implementation BookmarkButton(Private) | 501 @end |
| OLD | NEW |