Index: ui/aura_shell/drag_drop_controller.cc |
diff --git a/ui/aura_shell/drag_drop_controller.cc b/ui/aura_shell/drag_drop_controller.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..c13e6a1ca4f05ba338b4dd96cfdca98a5f5adfe0 |
--- /dev/null |
+++ b/ui/aura_shell/drag_drop_controller.cc |
@@ -0,0 +1,154 @@ |
+// Copyright (c) 2011 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "ui/aura_shell/drag_drop_controller.h" |
+ |
+#include "base/message_loop.h" |
+#include "ui/aura/desktop.h" |
+#include "ui/aura/window.h" |
+#include "ui/aura/window_drag_drop_delegate.h" |
+#include "ui/base/dragdrop/drag_drop_types.h" |
+#include "ui/base/dragdrop/os_exchange_data_provider_aura.h" |
+#include "ui/gfx/point.h" |
+#include "ui/gfx/rect.h" |
+#include "views/controls/image_view.h" |
+#include "views/widget/widget.h" |
+ |
+namespace aura_shell { |
+namespace internal { |
+ |
+using aura::Desktop; |
+ |
+namespace { |
+using views::Widget; |
+ |
+Widget* CreateDragWidget() { |
Ben Goodger (Google)
2011/11/11 17:26:53
I'd like it if the whole "view" part of the drag&d
varunjain
2011/11/15 19:39:33
Done.
|
+ Widget* drag_widget = new Widget; |
+ Widget::InitParams params; |
+ params.type = Widget::InitParams::TYPE_TOOLTIP; |
+ params.keep_on_top = true; |
+ params.accept_events = false; |
+ params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; |
+ params.transparent = true; |
+ drag_widget->Init(params); |
+ drag_widget->SetOpacity(0xFF); |
+ return drag_widget; |
+} |
+ |
+} // namespace |
+ |
+DragDropController::DragDropController() |
+ : drag_widget_(NULL), |
+ drag_data_(NULL), |
+ drag_operation_(0), |
+ dragged_window_(NULL), |
+ drag_drop_in_progress_(false), |
+ should_block_during_drag_drop_(true) { |
+} |
+ |
+DragDropController::~DragDropController() { |
+ Cleanup(); |
+} |
+ |
+void DragDropController::StartDragAndDrop(const ui::OSExchangeData& data, |
+ int operation) { |
+ DCHECK(!drag_drop_in_progress_); |
+ aura::Window* capture_window = Desktop::GetInstance()->capture_window(); |
+ if (capture_window) |
+ Desktop::GetInstance()->ReleaseCapture(capture_window); |
+ drag_drop_in_progress_ = true; |
+ |
+ DCHECK(drag_data_ == NULL && drag_operation_ == 0); |
+ DCHECK(drag_widget_.get() == NULL); |
+ drag_data_ = &data; |
+ drag_widget_.reset(CreateDragWidget()); |
Ben Goodger (Google)
2011/11/11 17:26:53
This would become something like
drag_image_.res
varunjain
2011/11/15 19:39:33
Done.
|
+ drag_operation_ = operation; |
+ gfx::Point location = Desktop::GetInstance()->last_mouse_location(); |
+ const ui::OSExchangeDataProviderAura& provider = |
+ static_cast<const ui::OSExchangeDataProviderAura&>(data.provider()); |
+ |
+ views::ImageView* image_view = new views::ImageView(); |
+ image_view->SetImage(provider.drag_image()); |
+ drag_widget_->SetContentsView(image_view); |
+ drag_widget_->SetBounds(gfx::Rect(location, |
+ image_view->GetPreferredSize())); |
+ drag_widget_->Show(); |
+ |
+ dragged_window_ = Desktop::GetInstance()->GetEventHandlerForPoint(location); |
+ |
+ if (should_block_during_drag_drop_) { |
+ MessageLoopForUI::current()->RunWithDispatcher( |
+ Desktop::GetInstance()->GetDispatcher()); |
+ } |
+} |
+ |
+void DragDropController::DragUpdate(aura::Window* target, |
+ const aura::MouseEvent& event) { |
+ if (target != dragged_window_) { |
+ if (dragged_window_ && dragged_window_->drag_drop_delegate()) |
+ dragged_window_->drag_drop_delegate()->OnDragExited(); |
+ |
+ dragged_window_ = target; |
+ if (dragged_window_ && dragged_window_->drag_drop_delegate()) { |
+ aura::DropTargetEvent e(*drag_data_, event.location(), drag_operation_); |
+ if (dragged_window_->drag_drop_delegate()->CanDrop(e)) |
+ dragged_window_->drag_drop_delegate()->OnDragEntered(e); |
+ } |
+ } else { |
+ if (dragged_window_ && dragged_window_->drag_drop_delegate()) { |
+ aura::DropTargetEvent e(*drag_data_, event.location(), drag_operation_); |
+ dragged_window_->drag_drop_delegate()->OnDragUpdated(e); |
+ // TODO(varunjain): uncomment the following lines when cursor issue with |
+ // X for tests is fixed. |
+ // gfx::NativeCursor cursor = (op == ui::DragDropTypes::DRAG_NONE)? |
+ // aura::kCursorMove : aura::kCursorHand; |
+ // Desktop::GetInstance()->SetCursor(cursor); |
+ } |
+ } |
+ |
+ DCHECK(drag_widget_.get()); |
+ if (drag_widget_->IsVisible()) { |
+ gfx::Rect bounds = drag_widget_->GetClientAreaScreenBounds(); |
+ bounds.set_origin(Desktop::GetInstance()->last_mouse_location()); |
+ drag_widget_->SetBounds(bounds); |
Ben Goodger (Google)
2011/11/11 17:26:53
And this would become drag_image_->SetPosition(Des
varunjain
2011/11/15 19:39:33
Done.
|
+ } |
+} |
+ |
+void DragDropController::Drop(aura::Window* target, |
+ const aura::MouseEvent& event) { |
+ DCHECK(target == dragged_window_); |
+ if (dragged_window_ && dragged_window_->drag_drop_delegate() ) { |
+ aura::DropTargetEvent e(*drag_data_, event.location(), drag_operation_); |
+ if (dragged_window_->drag_drop_delegate()->CanDrop(e)) |
+ dragged_window_->drag_drop_delegate()->OnPerformDrop(e); |
+ // TODO(varunjain): else Do drag widget flying back animation |
+ } |
+ |
+ Cleanup(); |
+ if (should_block_during_drag_drop_) |
+ MessageLoop::current()->Quit(); |
+} |
+ |
+void DragDropController::DragCancel() { |
+ // TODO(varunjain): Do drag widget flying back animation |
+ Cleanup(); |
+ if (should_block_during_drag_drop_) |
+ MessageLoop::current()->Quit(); |
+} |
+ |
+bool DragDropController::IsDragAndDropInProgress() { |
+ return drag_drop_in_progress_; |
+} |
+ |
+void DragDropController::Cleanup() { |
+ if (drag_widget_.get()) |
+ drag_widget_->Hide(); |
Ben Goodger (Google)
2011/11/11 17:26:53
This should all be done by drag_image_'s dtor.
varunjain
2011/11/15 19:39:33
Done.
|
+ drag_widget_.reset(); |
+ drag_data_ = NULL; |
+ drag_operation_ = 0; |
+ drag_drop_in_progress_ = false; |
+} |
+ |
+} // namespace internal |
+} // namespace aura_shell |