summaryrefslogtreecommitdiffstats
path: root/app/win
diff options
context:
space:
mode:
authorbrettw@chromium.org <brettw@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-10-17 20:44:38 +0000
committerbrettw@chromium.org <brettw@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-10-17 20:44:38 +0000
commit79a4c1e532a74ac74d72f883e52e30019fe9e731 (patch)
tree2fc9915dc6e1fe9c4b9bea3bdda5ef211da49980 /app/win
parent13147160ae28933269024ba973f0a1ae3d11a57d (diff)
downloadchromium_src-79a4c1e532a74ac74d72f883e52e30019fe9e731.zip
chromium_src-79a4c1e532a74ac74d72f883e52e30019fe9e731.tar.gz
chromium_src-79a4c1e532a74ac74d72f883e52e30019fe9e731.tar.bz2
Move BaseDropTarget and BaseDragSource from base to app/win. Remove the "Base" class name prefix and put in the app::win namespace.
TEST=none BUG=none Review URL: http://codereview.chromium.org/3822007 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@62900 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'app/win')
-rw-r--r--app/win/drag_source.cc57
-rw-r--r--app/win/drag_source.h58
-rw-r--r--app/win/drop_target.cc174
-rw-r--r--app/win/drop_target.h136
4 files changed, 425 insertions, 0 deletions
diff --git a/app/win/drag_source.cc b/app/win/drag_source.cc
new file mode 100644
index 0000000..a695a33
--- /dev/null
+++ b/app/win/drag_source.cc
@@ -0,0 +1,57 @@
+// Copyright (c) 2010 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 "app/win/drag_source.h"
+
+namespace app {
+namespace win {
+
+DragSource::DragSource() : cancel_drag_(false) {
+}
+
+HRESULT DragSource::QueryContinueDrag(BOOL escape_pressed, DWORD key_state) {
+ if (cancel_drag_)
+ return DRAGDROP_S_CANCEL;
+
+ if (escape_pressed) {
+ OnDragSourceCancel();
+ return DRAGDROP_S_CANCEL;
+ }
+
+ if (!(key_state & MK_LBUTTON)) {
+ OnDragSourceDrop();
+ return DRAGDROP_S_DROP;
+ }
+
+ OnDragSourceMove();
+ return S_OK;
+}
+
+HRESULT DragSource::GiveFeedback(DWORD effect) {
+ return DRAGDROP_S_USEDEFAULTCURSORS;
+}
+
+HRESULT DragSource::QueryInterface(const IID& iid, void** object) {
+ *object = NULL;
+ if (IsEqualIID(iid, IID_IUnknown) || IsEqualIID(iid, IID_IDropSource)) {
+ *object = this;
+ } else {
+ return E_NOINTERFACE;
+ }
+ AddRef();
+ return S_OK;
+}
+
+ULONG DragSource::AddRef() {
+ base::RefCountedThreadSafe<DragSource>::AddRef();
+ return 0;
+}
+
+ULONG DragSource::Release() {
+ base::RefCountedThreadSafe<DragSource>::Release();
+ return 0;
+}
+
+} // namespace win
+} // namespace app
diff --git a/app/win/drag_source.h b/app/win/drag_source.h
new file mode 100644
index 0000000..75bee86
--- /dev/null
+++ b/app/win/drag_source.h
@@ -0,0 +1,58 @@
+// Copyright (c) 2010 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.
+
+#ifndef APP_WIN_DRAG_SOURCE_H_
+#define APP_WIN_DRAG_SOURCE_H_
+#pragma once
+
+#include <objidl.h>
+
+#include "base/basictypes.h"
+#include "base/ref_counted.h"
+
+namespace app {
+namespace win {
+
+// A base IDropSource implementation. Handles notifications sent by an active
+// drag-drop operation as the user mouses over other drop targets on their
+// system. This object tells Windows whether or not the drag should continue,
+// and supplies the appropriate cursors.
+class DragSource : public IDropSource,
+ public base::RefCountedThreadSafe<DragSource> {
+ public:
+ DragSource();
+ virtual ~DragSource() {}
+
+ // Stop the drag operation at the next chance we get. This doesn't
+ // synchronously stop the drag (since Windows is controlling that),
+ // but lets us tell Windows to cancel the drag the next chance we get.
+ void CancelDrag() {
+ cancel_drag_ = true;
+ }
+
+ // IDropSource implementation:
+ HRESULT __stdcall QueryContinueDrag(BOOL escape_pressed, DWORD key_state);
+ HRESULT __stdcall GiveFeedback(DWORD effect);
+
+ // IUnknown implementation:
+ HRESULT __stdcall QueryInterface(const IID& iid, void** object);
+ ULONG __stdcall AddRef();
+ ULONG __stdcall Release();
+
+ protected:
+ virtual void OnDragSourceCancel() {}
+ virtual void OnDragSourceDrop() {}
+ virtual void OnDragSourceMove() {}
+
+ private:
+ // Set to true if we want to cancel the drag operation.
+ bool cancel_drag_;
+
+ DISALLOW_COPY_AND_ASSIGN(DragSource);
+};
+
+} // namespace win
+} // namespace app
+
+#endif // APP_WIN_DRAG_SOURCE_H_
diff --git a/app/win/drop_target.cc b/app/win/drop_target.cc
new file mode 100644
index 0000000..6742e19
--- /dev/null
+++ b/app/win/drop_target.cc
@@ -0,0 +1,174 @@
+// Copyright (c) 2010 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 "app/win/drop_target.h"
+
+#include <shlobj.h>
+
+#include "base/logging.h"
+
+namespace app {
+namespace win {
+
+IDropTargetHelper* DropTarget::cached_drop_target_helper_ = NULL;
+int32 DropTarget::drag_identity_ = 0;
+
+DropTarget::DropTarget(HWND hwnd)
+ : hwnd_(hwnd),
+ suspended_(false),
+ ref_count_(0) {
+ DCHECK(hwnd);
+ HRESULT result = RegisterDragDrop(hwnd, this);
+ DCHECK(SUCCEEDED(result));
+}
+
+DropTarget::~DropTarget() {
+}
+
+// static
+IDropTargetHelper* DropTarget::DropHelper() {
+ if (!cached_drop_target_helper_) {
+ CoCreateInstance(CLSID_DragDropHelper, 0, CLSCTX_INPROC_SERVER,
+ IID_IDropTargetHelper,
+ reinterpret_cast<void**>(&cached_drop_target_helper_));
+ }
+ return cached_drop_target_helper_;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+// DropTarget, IDropTarget implementation:
+
+HRESULT DropTarget::DragEnter(IDataObject* data_object,
+ DWORD key_state,
+ POINTL cursor_position,
+ DWORD* effect) {
+ // Tell the helper that we entered so it can update the drag image.
+ IDropTargetHelper* drop_helper = DropHelper();
+ if (drop_helper) {
+ drop_helper->DragEnter(GetHWND(), data_object,
+ reinterpret_cast<POINT*>(&cursor_position), *effect);
+ }
+
+ // You can't drag and drop within the same HWND.
+ if (suspended_) {
+ *effect = DROPEFFECT_NONE;
+ return S_OK;
+ }
+
+ // Update the drag identity, skipping 0.
+ if (++drag_identity_ == 0)
+ ++drag_identity_;
+
+ current_data_object_ = data_object;
+ POINT screen_pt = { cursor_position.x, cursor_position.y };
+ *effect = OnDragEnter(current_data_object_, key_state, screen_pt, *effect);
+ return S_OK;
+}
+
+HRESULT DropTarget::DragOver(DWORD key_state,
+ POINTL cursor_position,
+ DWORD* effect) {
+ // Tell the helper that we moved over it so it can update the drag image.
+ IDropTargetHelper* drop_helper = DropHelper();
+ if (drop_helper)
+ drop_helper->DragOver(reinterpret_cast<POINT*>(&cursor_position), *effect);
+
+ if (suspended_) {
+ *effect = DROPEFFECT_NONE;
+ return S_OK;
+ }
+
+ POINT screen_pt = { cursor_position.x, cursor_position.y };
+ *effect = OnDragOver(current_data_object_, key_state, screen_pt, *effect);
+ return S_OK;
+}
+
+HRESULT DropTarget::DragLeave() {
+ // Tell the helper that we moved out of it so it can update the drag image.
+ IDropTargetHelper* drop_helper = DropHelper();
+ if (drop_helper)
+ drop_helper->DragLeave();
+
+ if (suspended_)
+ return S_OK;
+
+ OnDragLeave(current_data_object_);
+
+ current_data_object_ = NULL;
+ return S_OK;
+}
+
+HRESULT DropTarget::Drop(IDataObject* data_object,
+ DWORD key_state,
+ POINTL cursor_position,
+ DWORD* effect) {
+ // Tell the helper that we dropped onto it so it can update the drag image.
+ IDropTargetHelper* drop_helper = DropHelper();
+ if (drop_helper) {
+ drop_helper->Drop(current_data_object_,
+ reinterpret_cast<POINT*>(&cursor_position), *effect);
+ }
+
+ if (suspended_) {
+ *effect = DROPEFFECT_NONE;
+ return S_OK;
+ }
+
+ POINT screen_pt = { cursor_position.x, cursor_position.y };
+ *effect = OnDrop(current_data_object_, key_state, screen_pt, *effect);
+ return S_OK;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+// DropTarget, IUnknown implementation:
+
+HRESULT DropTarget::QueryInterface(const IID& iid, void** object) {
+ *object = NULL;
+ if (IsEqualIID(iid, IID_IUnknown) || IsEqualIID(iid, IID_IDropTarget)) {
+ *object = this;
+ } else {
+ return E_NOINTERFACE;
+ }
+ AddRef();
+ return S_OK;
+}
+
+ULONG DropTarget::AddRef() {
+ return ++ref_count_;
+}
+
+ULONG DropTarget::Release() {
+ if (--ref_count_ == 0) {
+ delete this;
+ return 0U;
+ }
+ return ref_count_;
+}
+
+DWORD DropTarget::OnDragEnter(IDataObject* data_object,
+ DWORD key_state,
+ POINT cursor_position,
+ DWORD effect) {
+ return DROPEFFECT_NONE;
+}
+
+DWORD DropTarget::OnDragOver(IDataObject* data_object,
+ DWORD key_state,
+ POINT cursor_position,
+ DWORD effect) {
+ return DROPEFFECT_NONE;
+}
+
+void DropTarget::OnDragLeave(IDataObject* data_object) {
+}
+
+DWORD DropTarget::OnDrop(IDataObject* data_object,
+ DWORD key_state,
+ POINT cursor_position,
+ DWORD effect) {
+ return DROPEFFECT_NONE;
+}
+
+} // namespace win
+} // namespace app
diff --git a/app/win/drop_target.h b/app/win/drop_target.h
new file mode 100644
index 0000000..e7dc1ec
--- /dev/null
+++ b/app/win/drop_target.h
@@ -0,0 +1,136 @@
+// Copyright (c) 2010 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.
+
+#ifndef APP_WIN_DROP_TARGET_H_
+#define APP_WIN_DROP_TARGET_H_
+#pragma once
+
+#include <objidl.h>
+
+#include "base/ref_counted.h"
+
+// Windows interface.
+struct IDropTargetHelper;
+
+namespace app {
+namespace win {
+
+// A DropTarget implementation that takes care of the nitty gritty
+// of dnd. While this class is concrete, subclasses will most likely
+// want to override various OnXXX methods.
+//
+// Because DropTarget is ref counted you shouldn't delete it directly,
+// rather wrap it in a scoped_refptr. Be sure and invoke RevokeDragDrop(m_hWnd)
+// before the HWND is deleted too.
+//
+// This class is meant to be used in a STA and is not multithread-safe.
+class DropTarget : public IDropTarget {
+ public:
+ // Create a new DropTarget associating it with the given HWND.
+ explicit DropTarget(HWND hwnd);
+ virtual ~DropTarget();
+
+ // When suspended is set to |true|, the drop target does not receive drops
+ // from drags initiated within the owning HWND.
+ bool suspended() const { return suspended_; }
+ void set_suspended(bool suspended) { suspended_ = suspended; }
+
+ // IDropTarget implementation:
+ HRESULT __stdcall DragEnter(IDataObject* data_object,
+ DWORD key_state,
+ POINTL cursor_position,
+ DWORD* effect);
+ HRESULT __stdcall DragOver(DWORD key_state,
+ POINTL cursor_position,
+ DWORD* effect);
+ HRESULT __stdcall DragLeave();
+ HRESULT __stdcall Drop(IDataObject* data_object,
+ DWORD key_state,
+ POINTL cursor_position,
+ DWORD* effect);
+
+ // IUnknown implementation:
+ HRESULT __stdcall QueryInterface(const IID& iid, void** object);
+ ULONG __stdcall AddRef();
+ ULONG __stdcall Release();
+
+ protected:
+ // Returns the hosting HWND.
+ HWND GetHWND() { return hwnd_; }
+
+ // Invoked when the cursor first moves over the hwnd during a dnd session.
+ // This should return a bitmask of the supported drop operations:
+ // DROPEFFECT_NONE, DROPEFFECT_COPY, DROPEFFECT_LINK and/or
+ // DROPEFFECT_MOVE.
+ virtual DWORD OnDragEnter(IDataObject* data_object,
+ DWORD key_state,
+ POINT cursor_position,
+ DWORD effect);
+
+ // Invoked when the cursor moves over the window during a dnd session.
+ // This should return a bitmask of the supported drop operations:
+ // DROPEFFECT_NONE, DROPEFFECT_COPY, DROPEFFECT_LINK and/or
+ // DROPEFFECT_MOVE.
+ virtual DWORD OnDragOver(IDataObject* data_object,
+ DWORD key_state,
+ POINT cursor_position,
+ DWORD effect);
+
+ // Invoked when the cursor moves outside the bounds of the hwnd during a
+ // dnd session.
+ virtual void OnDragLeave(IDataObject* data_object);
+
+ // Invoked when the drop ends on the window. This should return the operation
+ // that was taken.
+ virtual DWORD OnDrop(IDataObject* data_object,
+ DWORD key_state,
+ POINT cursor_position,
+ DWORD effect);
+
+ // Return the drag identity.
+ static int32 GetDragIdentity() { return drag_identity_; }
+
+ private:
+ // Returns the cached drop helper, creating one if necessary. The returned
+ // object is not addrefed. May return NULL if the object couldn't be created.
+ static IDropTargetHelper* DropHelper();
+
+ // The data object currently being dragged over this drop target.
+ scoped_refptr<IDataObject> current_data_object_;
+
+ // A helper object that is used to provide drag image support while the mouse
+ // is dragging over the content area.
+ //
+ // DO NOT ACCESS DIRECTLY! Use DropHelper() instead, which will lazily create
+ // this if it doesn't exist yet. This object can take tens of milliseconds to
+ // create, and we don't want to block any window opening for this, especially
+ // since often, DnD will never be used. Instead, we force this penalty to the
+ // first time it is actually used.
+ static IDropTargetHelper* cached_drop_target_helper_;
+
+ // The drag identity (id). An up-counter that increases when the cursor first
+ // moves over the HWND in a DnD session (OnDragEnter). 0 is reserved to mean
+ // the "no/unknown" identity, and is used for initialization. The identity is
+ // sent to the renderer in drag enter notifications. Note: the identity value
+ // is passed over the renderer NPAPI interface to gears, so use int32 instead
+ // of int here.
+ static int32 drag_identity_;
+
+ // The HWND of the source. This HWND is used to determine coordinates for
+ // mouse events that are sent to the renderer notifying various drag states.
+ HWND hwnd_;
+
+ // Whether or not we are currently processing drag notifications for drags
+ // initiated in this window.
+ bool suspended_;
+
+ LONG ref_count_;
+
+ DISALLOW_COPY_AND_ASSIGN(DropTarget);
+};
+
+} // namespace win
+} // namespace app
+
+#endif // APP_WIN_DROP_TARGET_H_