Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(667)

Unified Diff: ui/views/cocoa/drag_drop_client_mac.mm

Issue 1964283002: MacViews: Implemented Drag & Drop (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: fix for dcheng Created 4 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « ui/views/cocoa/drag_drop_client_mac.h ('k') | ui/views/cocoa/drag_drop_client_mac_unittest.mm » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: ui/views/cocoa/drag_drop_client_mac.mm
diff --git a/ui/views/cocoa/drag_drop_client_mac.mm b/ui/views/cocoa/drag_drop_client_mac.mm
new file mode 100644
index 0000000000000000000000000000000000000000..737a0a0e4b0ce88521a765225da44717bff18179
--- /dev/null
+++ b/ui/views/cocoa/drag_drop_client_mac.mm
@@ -0,0 +1,163 @@
+// Copyright 2016 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import "ui/views/cocoa/drag_drop_client_mac.h"
+
+#include "base/mac/mac_util.h"
+#include "base/run_loop.h"
+#include "base/strings/sys_string_conversions.h"
+#import "ui/base/dragdrop/os_exchange_data_provider_mac.h"
+#include "ui/gfx/image/image_skia_util_mac.h"
+#include "ui/views/drag_utils.h"
+#import "ui/views/cocoa/bridged_content_view.h"
+#import "ui/views/cocoa/bridged_native_widget.h"
+#include "ui/views/widget/native_widget_mac.h"
+
+@interface CocoaDragDropDataProvider ()
+- (id)initWithData:(const ui::OSExchangeData&)data;
+- (id)initWithPasteboard:(NSPasteboard*)pasteboard;
+@end
+
+@implementation CocoaDragDropDataProvider
+
+std::unique_ptr<ui::OSExchangeData> data_;
Scott Hess - ex-Googler 2016/06/05 20:11:51 Dollars to donuts this is responsible for the wate
tapted 2016/06/05 23:37:35 Ah doh - it just needs curlies around it. I'll sub
+
+- (id)initWithData:(const ui::OSExchangeData&)data {
+ if ((self = [super init])) {
+ data_.reset(new OSExchangeData(data.provider().Clone()));
+ }
+ return self;
+}
+
+- (id)initWithPasteboard:(NSPasteboard*)pasteboard {
+ if ((self = [super init])) {
+ data_ = ui::OSExchangeDataProviderMac::CreateDataFromPasteboard(pasteboard);
+ }
+ return self;
+}
+
+- (ui::OSExchangeData*)data {
+ return data_.get();
+}
+
+// NSPasteboardItemDataProvider protocol implementation.
+
+- (void)pasteboard:(NSPasteboard*)sender
+ item:(NSPasteboardItem*)item
+ provideDataForType:(NSString*)type {
+ const ui::OSExchangeDataProviderMac& provider =
+ static_cast<const ui::OSExchangeDataProviderMac&>(data_->provider());
+ NSData* ns_data = provider.GetNSDataForType(type);
+ [sender setData:ns_data forType:type];
+}
+
+@end
+
+namespace views {
+
+DragDropClientMac::DragDropClientMac(BridgedNativeWidget* bridge,
+ View* root_view)
+ : drop_helper_(root_view),
+ operation_(0),
+ bridge_(bridge),
+ quit_closure_(base::Closure()) {
+ DCHECK(bridge);
+}
+
+DragDropClientMac::~DragDropClientMac() {}
+
+void DragDropClientMac::StartDragAndDrop(
+ View* view,
+ const ui::OSExchangeData& data,
+ int operation,
+ ui::DragDropTypes::DragEventSource source) {
+ data_source_.reset([[CocoaDragDropDataProvider alloc] initWithData:data]);
+ operation_ = operation;
+
+ const ui::OSExchangeDataProviderMac& provider =
+ static_cast<const ui::OSExchangeDataProviderMac&>(data.provider());
+
+ // Synthesize an event for dragging, since we can't be sure that
+ // [NSApp currentEvent] will return a valid dragging event.
+ NSWindow* window = bridge_->ns_window();
+ NSPoint position = [window mouseLocationOutsideOfEventStream];
+ NSTimeInterval event_time = [[NSApp currentEvent] timestamp];
+ NSEvent* event = [NSEvent mouseEventWithType:NSLeftMouseDragged
+ location:position
+ modifierFlags:NSLeftMouseDraggedMask
+ timestamp:event_time
+ windowNumber:[window windowNumber]
+ context:nil
+ eventNumber:0
+ clickCount:1
+ pressure:1.0];
+
+ NSImage* image = gfx::NSImageFromImageSkiaWithColorSpace(
+ provider.GetDragImage(), base::mac::GetSRGBColorSpace());
+
+ base::scoped_nsobject<NSPasteboardItem> item([[NSPasteboardItem alloc] init]);
+ [item setDataProvider:data_source_.get()
+ forTypes:ui::OSExchangeDataProviderMac::
+ SupportedPasteboardTypes()];
+
+ base::scoped_nsobject<NSDraggingItem> drag_item(
+ [[NSDraggingItem alloc] initWithPasteboardWriter:item.get()]);
+
+ // Subtract the image's height from the y location so that the mouse will be
+ // at the upper left corner of the image.
+ NSRect dragging_frame =
+ NSMakeRect([event locationInWindow].x,
+ [event locationInWindow].y - [image size].height,
+ [image size].width, [image size].height);
+ [drag_item setDraggingFrame:dragging_frame contents:image];
+
+ [bridge_->ns_view() beginDraggingSessionWithItems:@[ drag_item.get() ]
+ event:event
+ source:bridge_->ns_view()];
+
+ // Since Drag and drop is asynchronous on Mac, we need to spin a nested run
+ // loop for consistency with other platforms.
+ base::RunLoop run_loop;
+ quit_closure_ = run_loop.QuitClosure();
+ run_loop.Run();
+}
+
+NSDragOperation DragDropClientMac::DragUpdate(id<NSDraggingInfo> sender) {
+ int drag_operation = ui::DragDropTypes::DRAG_NONE;
+
+ // Since dragging from non MacView sources does not generate OSExchangeData,
+ // we need to generate one based on the provided pasteboard.
+ if (!data_source_.get()) {
+ data_source_.reset([[CocoaDragDropDataProvider alloc]
+ initWithPasteboard:[sender draggingPasteboard]]);
+ }
+
+ drag_operation = drop_helper_.OnDragOver(
+ *[data_source_ data], LocationInView([sender draggingLocation]),
+ operation_);
+ return ui::DragDropTypes::DragOperationToNSDragOperation(drag_operation);
+}
+
+NSDragOperation DragDropClientMac::Drop(id<NSDraggingInfo> sender) {
+ int drag_operation = drop_helper_.OnDrop(
+ *[data_source_ data], LocationInView([sender draggingLocation]),
+ operation_);
+ return ui::DragDropTypes::DragOperationToNSDragOperation(drag_operation);
+}
+
+void DragDropClientMac::EndDrag() {
+ data_source_.reset();
+
+ // Allow a test to invoke EndDrag() without spinning the nested run loop.
+ if (!quit_closure_.is_null()) {
+ quit_closure_.Run();
+ quit_closure_.Reset();
+ }
+}
+
+gfx::Point DragDropClientMac::LocationInView(NSPoint point) const {
+ return gfx::Point(point.x, NSHeight([bridge_->ns_window() frame]) - point.y);
+}
+
+} // namespace views
« no previous file with comments | « ui/views/cocoa/drag_drop_client_mac.h ('k') | ui/views/cocoa/drag_drop_client_mac_unittest.mm » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698