| Index: ui/views/widget/desktop_aura/desktop_drag_drop_client_aurax11.cc
|
| diff --git a/ui/views/widget/desktop_aura/desktop_drag_drop_client_aurax11.cc b/ui/views/widget/desktop_aura/desktop_drag_drop_client_aurax11.cc
|
| index d3e52f787192b6ec1a11eebd26d7e8843e204637..531718207b0c9a54287c03d7ad6f5154552531a2 100644
|
| --- a/ui/views/widget/desktop_aura/desktop_drag_drop_client_aurax11.cc
|
| +++ b/ui/views/widget/desktop_aura/desktop_drag_drop_client_aurax11.cc
|
| @@ -19,8 +19,11 @@
|
| #include "ui/base/x/x11_util.h"
|
| #include "ui/events/event.h"
|
| #include "ui/events/platform/platform_event_source.h"
|
| +#include "ui/gfx/screen.h"
|
| +#include "ui/views/controls/image_view.h"
|
| #include "ui/views/widget/desktop_aura/desktop_native_cursor_manager.h"
|
| #include "ui/views/widget/desktop_aura/x11_topmost_window_finder.h"
|
| +#include "ui/views/widget/widget.h"
|
| #include "ui/wm/public/drag_drop_client.h"
|
| #include "ui/wm/public/drag_drop_delegate.h"
|
|
|
| @@ -29,6 +32,11 @@ using ui::OSExchangeData;
|
|
|
| namespace {
|
|
|
| +// The minimum alpha before we declare a pixel transparent when searching in
|
| +// our source image.
|
| +const uint32 kMinAlpha = 32;
|
| +const unsigned char kDragWidgetOpacity = 0xc0;
|
| +
|
| const int kMinXdndVersion = 5;
|
|
|
| const int kWillAcceptDrop = 1;
|
| @@ -80,6 +88,25 @@ static base::LazyInstance<
|
| std::map< ::Window, views::DesktopDragDropClientAuraX11*> >::Leaky
|
| g_live_client_map = LAZY_INSTANCE_INITIALIZER;
|
|
|
| +// Checks to see if |drag_image| is an image that has any visible regions
|
| +// (defined as having a pixel with alpha > 32). If so, returns true.
|
| +bool CheckIfIconValid(const gfx::ImageSkia& drag_image) {
|
| + // Because we need a GL context per window, we do a quick check if the window
|
| + // would just be displaying a mostly transparent image.
|
| + const SkBitmap* in_bitmap = drag_image.bitmap();
|
| + SkAutoLockPixels in_lock(*in_bitmap);
|
| + for (int y = 0; y < in_bitmap->height(); ++y) {
|
| + uint32* in_row = in_bitmap->getAddr32(0, y);
|
| +
|
| + for (int x = 0; x < in_bitmap->width(); ++x) {
|
| + if (SkColorGetA(in_row[x]) > kMinAlpha)
|
| + return true;
|
| + }
|
| + }
|
| +
|
| + return false;
|
| +}
|
| +
|
| } // namespace
|
|
|
| namespace views {
|
| @@ -363,6 +390,7 @@ DesktopDragDropClientAuraX11::DesktopDragDropClientAuraX11(
|
| grab_cursor_(cursor_manager->GetInitializedCursor(ui::kCursorGrabbing)),
|
| copy_grab_cursor_(cursor_manager->GetInitializedCursor(ui::kCursorCopy)),
|
| move_grab_cursor_(cursor_manager->GetInitializedCursor(ui::kCursorMove)),
|
| + null_drag_widget_bounds_(-100, -100, 1, 1),
|
| weak_ptr_factory_(this) {
|
| // Some tests change the DesktopDragDropClientAuraX11 associated with an
|
| // |xwindow|.
|
| @@ -477,17 +505,7 @@ void DesktopDragDropClientAuraX11::OnXdndStatus(
|
| return;
|
| }
|
|
|
| - switch (negotiated_operation_) {
|
| - case ui::DragDropTypes::DRAG_COPY:
|
| - move_loop_.UpdateCursor(copy_grab_cursor_);
|
| - break;
|
| - case ui::DragDropTypes::DRAG_MOVE:
|
| - move_loop_.UpdateCursor(move_grab_cursor_);
|
| - break;
|
| - default:
|
| - move_loop_.UpdateCursor(grab_cursor_);
|
| - break;
|
| - }
|
| + UpdateCursor();
|
|
|
| // Note: event.data.[2,3] specify a rectangle. It is a request by the other
|
| // window to not send further XdndPosition messages while the cursor is
|
| @@ -610,15 +628,21 @@ int DesktopDragDropClientAuraX11::StartDragAndDrop(
|
| base::WeakPtr<DesktopDragDropClientAuraX11> alive(
|
| weak_ptr_factory_.GetWeakPtr());
|
|
|
| + SetDragImage(source_provider_->GetDragImage(),
|
| + source_provider_->GetDragImageOffset());
|
| + CreateDragImageWindow();
|
| + // Capture drag widget to continue receiving events even if the drag source
|
| + // gets destroyed during the drag such as when we drag an item out of a menu
|
| + // and the menu closes.
|
| + drag_widget_->GetNativeWindow()->SetCapture();
|
| +
|
| // Windows has a specific method, DoDragDrop(), which performs the entire
|
| // drag. We have to emulate this, so we spin off a nested runloop which will
|
| // track all cursor movement and reroute events to a specific handler.
|
| - move_loop_.SetDragImage(source_provider_->GetDragImage(),
|
| - source_provider_->GetDragImageOffset());
|
| - move_loop_.RunMoveLoop(source_window, grab_cursor_);
|
| + move_loop_.RunMoveLoop();
|
|
|
| if (alive) {
|
| - move_loop_.SetDragImage(gfx::ImageSkia(), gfx::Vector2dF());
|
| + SetDragImage(gfx::ImageSkia(), gfx::Vector2dF());
|
|
|
| source_provider_ = NULL;
|
| g_current_drag_drop_client = NULL;
|
| @@ -702,6 +726,7 @@ void DesktopDragDropClientAuraX11::OnMouseReleased() {
|
| }
|
|
|
| void DesktopDragDropClientAuraX11::OnMoveLoopEnded() {
|
| + drag_widget_.reset();
|
| if (source_current_window_ != None) {
|
| SendXdndLeave(source_current_window_);
|
| source_current_window_ = None;
|
| @@ -779,6 +804,17 @@ void DesktopDragDropClientAuraX11::ProcessMouseMove(
|
| if (source_state_ != SOURCE_STATE_OTHER)
|
| return;
|
|
|
| + if (drag_widget_.get()) {
|
| + if (drag_image_.isNull()) {
|
| + drag_widget_->SetBounds(null_drag_widget_bounds_);
|
| + } else {
|
| + drag_widget_->SetBounds(gfx::Rect(
|
| + gfx::ToFlooredPoint(screen_point - drag_offset_),
|
| + drag_image_.size()));
|
| + }
|
| + drag_widget_->StackAtTop();
|
| + }
|
| +
|
| // Find the current window the cursor is over.
|
| ::Window dest_window = FindWindowFor(screen_point);
|
|
|
| @@ -911,6 +947,27 @@ ui::SelectionFormatMap DesktopDragDropClientAuraX11::GetFormatMap() const {
|
| ui::SelectionFormatMap();
|
| }
|
|
|
| +void DesktopDragDropClientAuraX11::UpdateCursor() {
|
| + gfx::NativeCursor cursor = grab_cursor_;
|
| + switch (negotiated_operation_) {
|
| + case ui::DragDropTypes::DRAG_COPY:
|
| + cursor = copy_grab_cursor_;
|
| + break;
|
| + case ui::DragDropTypes::DRAG_MOVE:
|
| + cursor = move_grab_cursor_;
|
| + break;
|
| + default:
|
| + break;
|
| + }
|
| +
|
| + // Update the pointer with the new cursor if a grab is active.
|
| + XChangeActivePointerGrab(
|
| + xdisplay_,
|
| + ButtonPressMask | ButtonReleaseMask | PointerMotionMask,
|
| + cursor.platform(),
|
| + CurrentTime);
|
| +}
|
| +
|
| void DesktopDragDropClientAuraX11::CompleteXdndPosition(
|
| ::Window source_window,
|
| const gfx::Point& screen_point) {
|
| @@ -1028,4 +1085,53 @@ void DesktopDragDropClientAuraX11::SendXdndDrop(::Window dest_window) {
|
| SendXClientEvent(dest_window, &xev);
|
| }
|
|
|
| +void DesktopDragDropClientAuraX11::SetDragImage(const gfx::ImageSkia& image,
|
| + const gfx::Vector2dF& offset) {
|
| + drag_image_ = image;
|
| + drag_offset_ = offset;
|
| + // Reset the Y offset, so that the drag-image is always just below the cursor,
|
| + // so that it is possible to see where the cursor is going.
|
| + drag_offset_.set_y(0.f);
|
| +}
|
| +
|
| +void DesktopDragDropClientAuraX11::CreateDragImageWindow() {
|
| + Widget* widget = new Widget;
|
| + Widget::InitParams params(Widget::InitParams::TYPE_DRAG);
|
| + params.opacity = Widget::InitParams::TRANSLUCENT_WINDOW;
|
| + params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET;
|
| + params.accept_events = false;
|
| +
|
| + if (!drag_image_.isNull() && !CheckIfIconValid(drag_image_)) {
|
| + // Show drag widget off screen if the image is mostly transparent.
|
| + drag_image_ = gfx::ImageSkia();
|
| + drag_offset_ = gfx::Vector2dF();
|
| + }
|
| +
|
| + if (drag_image_.isNull()) {
|
| + params.bounds = null_drag_widget_bounds_;
|
| + } else {
|
| + params.bounds = gfx::Rect(
|
| + gfx::ToFlooredPoint(
|
| + gfx::Screen::GetNativeScreen()->GetCursorScreenPoint() -
|
| + drag_offset_),
|
| + drag_image_.size());
|
| + }
|
| + widget->set_focus_on_creation(false);
|
| + widget->set_frame_type(Widget::FRAME_TYPE_FORCE_NATIVE);
|
| + widget->Init(params);
|
| + widget->SetOpacity(kDragWidgetOpacity);
|
| + widget->GetNativeWindow()->SetName("DragWindow");
|
| +
|
| + if (!drag_image_.isNull()) {
|
| + ImageView* image = new ImageView();
|
| + image->SetImage(drag_image_);
|
| + image->SetBounds(0, 0, drag_image_.width(), drag_image_.height());
|
| + widget->SetContentsView(image);
|
| + }
|
| +
|
| + widget->Show();
|
| + widget->GetNativeWindow()->layer()->SetFillsBoundsOpaquely(false);
|
| + drag_widget_.reset(widget);
|
| +}
|
| +
|
| } // namespace views
|
|
|