293 lines
9.4 KiB
C++
293 lines
9.4 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is mozilla.org code.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Netscape Communications Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 1998
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "nsDragHelperService.h"
|
|
|
|
#include "nsGUIEvent.h"
|
|
#include "nsIDOMNode.h"
|
|
#include "nsIDragSessionMac.h"
|
|
#include "nsIServiceManager.h"
|
|
|
|
#define kDragServiceContractID "@mozilla.org/widget/dragservice;1"
|
|
|
|
|
|
NS_IMPL_ADDREF(nsDragHelperService)
|
|
NS_IMPL_RELEASE(nsDragHelperService)
|
|
NS_IMPL_QUERY_INTERFACE1(nsDragHelperService, nsIDragHelperService)
|
|
|
|
|
|
//
|
|
// nsDragHelperService constructor
|
|
//
|
|
nsDragHelperService::nsDragHelperService()
|
|
{
|
|
}
|
|
|
|
|
|
//
|
|
// nsDragHelperService destructor
|
|
//
|
|
nsDragHelperService::~nsDragHelperService()
|
|
{
|
|
NS_ASSERTION(!mDragService.get(),
|
|
"A drag was not correctly ended by shutdown");
|
|
}
|
|
|
|
|
|
//
|
|
// Enter
|
|
//
|
|
// Called when the mouse has entered the rectangle bounding the browser
|
|
// during a drag. Cache the drag service so we don't have to fetch it
|
|
// repeatedly, and handles setting up the drag reference and sending an
|
|
// enter event.
|
|
//
|
|
NS_IMETHODIMP
|
|
nsDragHelperService::Enter(DragReference inDragRef, nsIEventSink *inSink)
|
|
{
|
|
// get our drag service for the duration of the drag.
|
|
mDragService = do_GetService(kDragServiceContractID);
|
|
NS_ASSERTION(mDragService,
|
|
"Couldn't get a drag service, we're in biiig trouble");
|
|
if (!mDragService || !inSink)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
// tell the session about this drag
|
|
mDragService->StartDragSession();
|
|
nsCOMPtr<nsIDragSessionMac> macSession(do_QueryInterface(mDragService));
|
|
if (macSession)
|
|
macSession->SetDragReference(inDragRef);
|
|
|
|
// let gecko know that the mouse has entered the window so it
|
|
// can start tracking and sending enter/exit events to frames.
|
|
Point mouseLocGlobal;
|
|
::GetDragMouse(inDragRef, &mouseLocGlobal, nsnull);
|
|
PRBool handled = PR_FALSE;
|
|
inSink->DragEvent(NS_DRAGDROP_ENTER, mouseLocGlobal.h, mouseLocGlobal.v, 0L,
|
|
&handled);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
//
|
|
// Tracking
|
|
//
|
|
// Called while the mouse is inside the rectangle bounding the browser
|
|
// during a drag. The important thing done here is to clear the |canDrop|
|
|
// property of the drag session every time through. The event handlers
|
|
// will reset it if appropriate.
|
|
//
|
|
NS_IMETHODIMP
|
|
nsDragHelperService::Tracking(DragReference inDragRef, nsIEventSink *inSink,
|
|
PRBool* outDropAllowed)
|
|
{
|
|
NS_ASSERTION(mDragService,
|
|
"Couldn't get a drag service, we're in biiig trouble");
|
|
if (!mDragService || !inSink) {
|
|
*outDropAllowed = PR_FALSE;
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
Point mouseLocGlobal;
|
|
::GetDragMouse(inDragRef, &mouseLocGlobal, nsnull);
|
|
short modifiers;
|
|
::GetDragModifiers(inDragRef, &modifiers, nsnull, nsnull);
|
|
|
|
// set the drag action on the service so the frames know what is going on
|
|
SetDragActionBasedOnModifiers(modifiers);
|
|
|
|
// clear out the |canDrop| property of the drag session. If it's meant to
|
|
// be, it will be set again.
|
|
nsCOMPtr<nsIDragSession> session;
|
|
mDragService->GetCurrentSession(getter_AddRefs(session));
|
|
NS_ASSERTION(session, "If we don't have a drag session, we're fucked");
|
|
if (session)
|
|
session->SetCanDrop(PR_FALSE);
|
|
|
|
// pass into gecko for handling...
|
|
PRBool handled = PR_FALSE;
|
|
inSink->DragEvent(NS_DRAGDROP_OVER, mouseLocGlobal.h, mouseLocGlobal.v,
|
|
modifiers, &handled);
|
|
|
|
// check if gecko has since allowed the drop and return it
|
|
if (session)
|
|
session->GetCanDrop(outDropAllowed);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
//
|
|
// Leave
|
|
//
|
|
// Called when the mouse leaves the rectangle bounding the browser
|
|
// during a drag. Cleans up the drag service and releases it.
|
|
//
|
|
NS_IMETHODIMP
|
|
nsDragHelperService::Leave(DragReference inDragRef, nsIEventSink *inSink)
|
|
{
|
|
NS_ASSERTION(mDragService,
|
|
"Couldn't get a drag service, we're in biiig trouble");
|
|
if (!mDragService || !inSink)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
// clear out the dragRef in the drag session. We are guaranteed that
|
|
// this will be called _after_ the drop has been processed (if there
|
|
// is one), so we're not destroying valuable information if the drop
|
|
// was in our window.
|
|
nsCOMPtr<nsIDragSessionMac> macSession(do_QueryInterface(mDragService));
|
|
if (macSession)
|
|
macSession->SetDragReference(0);
|
|
|
|
// let gecko know that the mouse has left the window so it
|
|
// can stop tracking and sending enter/exit events to frames.
|
|
Point mouseLocGlobal;
|
|
::GetDragMouse(inDragRef, &mouseLocGlobal, nsnull);
|
|
PRBool handled = PR_FALSE;
|
|
inSink->DragEvent(NS_DRAGDROP_EXIT, mouseLocGlobal.h, mouseLocGlobal.v, 0L,
|
|
&handled);
|
|
|
|
#ifndef MOZ_WIDGET_COCOA
|
|
::HideDragHilite(inDragRef);
|
|
#endif
|
|
|
|
nsCOMPtr<nsIDragSession> currentDragSession;
|
|
mDragService->GetCurrentSession(getter_AddRefs(currentDragSession));
|
|
|
|
if (currentDragSession) {
|
|
nsCOMPtr<nsIDOMNode> sourceNode;
|
|
currentDragSession->GetSourceNode(getter_AddRefs(sourceNode));
|
|
|
|
if (!sourceNode) {
|
|
// We're leaving a window while doing a drag that was
|
|
// initiated in a differnt app. End the drag session,
|
|
// since we're done with it for now (until the user
|
|
// drags back into mozilla).
|
|
mDragService->EndDragSession();
|
|
}
|
|
}
|
|
|
|
// we're _really_ done with it, so let go of the service.
|
|
mDragService = nsnull;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
//
|
|
// Drop
|
|
//
|
|
// Called when a drop occurs within the rectangle bounding the browser
|
|
// during a drag. Cleans up the drag service and releases it.
|
|
//
|
|
NS_IMETHODIMP
|
|
nsDragHelperService::Drop(DragReference inDragRef, nsIEventSink *inSink,
|
|
PRBool* outAccepted)
|
|
{
|
|
NS_ASSERTION(mDragService,
|
|
"Couldn't get a drag service, we're in biiig trouble");
|
|
if (!mDragService || !inSink) {
|
|
*outAccepted = PR_FALSE;
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
// We make the assuption that the dragOver handlers have correctly set
|
|
// the |canDrop| property of the Drag Session. Before we dispatch the event
|
|
// into Gecko, check that value and either dispatch it or set the result
|
|
// code to "spring-back" and show the user the drag failed.
|
|
OSErr result = noErr;
|
|
nsCOMPtr<nsIDragSession> dragSession;
|
|
mDragService->GetCurrentSession(getter_AddRefs(dragSession));
|
|
if (dragSession) {
|
|
// if the target has set that it can accept the drag, pass along
|
|
// to gecko, otherwise set phasers for failure.
|
|
PRBool canDrop = PR_FALSE;
|
|
if (NS_SUCCEEDED(dragSession->GetCanDrop(&canDrop)))
|
|
if (canDrop) {
|
|
// pass the drop event along to Gecko
|
|
Point mouseLocGlobal;
|
|
::GetDragMouse(inDragRef, &mouseLocGlobal, nsnull);
|
|
short modifiers;
|
|
::GetDragModifiers(inDragRef, &modifiers, nsnull, nsnull);
|
|
PRBool handled = PR_FALSE;
|
|
inSink->DragEvent(NS_DRAGDROP_DROP, mouseLocGlobal.h, mouseLocGlobal.v,
|
|
modifiers, &handled);
|
|
}
|
|
else
|
|
result = dragNotAcceptedErr;
|
|
} // if a valid drag session
|
|
|
|
// we don't need the drag session anymore, the user has released the
|
|
// mouse and the event has already gone to gecko.
|
|
mDragService->EndDragSession();
|
|
|
|
// if there was any kind of error, the drag wasn't accepted
|
|
*outAccepted = (result == noErr);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
//
|
|
// SetDragActionsBasedOnModifiers
|
|
//
|
|
// Examines the MacOS modifier keys and sets the appropriate drag action on the
|
|
// drag session to copy/move/etc
|
|
//
|
|
void
|
|
nsDragHelperService::SetDragActionBasedOnModifiers(short inModifiers)
|
|
{
|
|
nsCOMPtr<nsIDragSession> dragSession;
|
|
mDragService->GetCurrentSession(getter_AddRefs(dragSession));
|
|
if (dragSession) {
|
|
PRUint32 action = nsIDragService::DRAGDROP_ACTION_MOVE;
|
|
|
|
// force copy = option, alias = cmd-option, default is move
|
|
if (inModifiers & optionKey) {
|
|
if (inModifiers & cmdKey)
|
|
action = nsIDragService::DRAGDROP_ACTION_LINK;
|
|
else
|
|
action = nsIDragService::DRAGDROP_ACTION_COPY;
|
|
}
|
|
|
|
dragSession->SetDragAction(action);
|
|
}
|
|
|
|
} // SetDragActionBasedOnModifiers
|
|
|