Files
libreoffice/vcl/source/window/dndevdis.cxx
Ras-al-Ghul f2f1f52d8b tdf#96888 Kill internal vcl dog-tags ...
Did a small change to dndevis.cxx file. Patch No. 1

Change-Id: I9240d97a9b0dd76a2df43e01f0d1e71de9015c48
Reviewed-on: https://gerrit.libreoffice.org/21211
Tested-by: Jenkins <ci@libreoffice.org>
Reviewed-by: Michael Meeks <michael.meeks@collabora.com>
2016-01-08 13:49:31 +00:00

474 lines
14 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0 .
*/
#include <dndevdis.hxx>
#include <dndlcon.hxx>
#include <window.h>
#include <svdata.hxx>
#include <osl/mutex.hxx>
#include <vcl/svapp.hxx>
#include <vcl/settings.hxx>
using namespace ::cppu;
using namespace ::com::sun::star::uno;
using namespace ::com::sun::star::lang;
using namespace ::com::sun::star::datatransfer;
using namespace ::com::sun::star::datatransfer::dnd;
// DNDEventDispatcher::DNDEventDispatcher
DNDEventDispatcher::DNDEventDispatcher( vcl::Window * pTopWindow ):
m_pTopWindow( pTopWindow ),
m_pCurrentWindow( nullptr )
{
}
// DNDEventDispatcher::~DNDEventDispatcher
DNDEventDispatcher::~DNDEventDispatcher()
{
designate_currentwindow(nullptr);
}
vcl::Window* DNDEventDispatcher::findTopLevelWindow(Point location)
{
SolarMutexGuard aSolarGuard;
// find the window that is toplevel for this coordinates
// because those coordinates come from outside, they must be mirrored if RTL layout is active
if( AllSettings::GetLayoutRTL() )
m_pTopWindow->ImplMirrorFramePos( location );
vcl::Window * pChildWindow = m_pTopWindow->ImplFindWindow( location );
if( nullptr == pChildWindow )
pChildWindow = m_pTopWindow;
while( pChildWindow->ImplGetClientWindow() )
pChildWindow = pChildWindow->ImplGetClientWindow();
if( pChildWindow->ImplIsAntiparallel() )
{
const OutputDevice *pChildWinOutDev = pChildWindow->GetOutDev();
pChildWinOutDev->ReMirror( location );
}
return pChildWindow;
}
IMPL_LINK_TYPED(DNDEventDispatcher, WindowEventListener, VclWindowEvent&, rEvent, void)
{
if (rEvent.GetId() == VCLEVENT_OBJECT_DYING)
{
designate_currentwindow(nullptr);
}
}
void DNDEventDispatcher::designate_currentwindow(vcl::Window *pWindow)
{
if (m_pCurrentWindow)
m_pCurrentWindow->RemoveEventListener(LINK(this, DNDEventDispatcher, WindowEventListener));
m_pCurrentWindow = pWindow;
if (m_pCurrentWindow)
m_pCurrentWindow->AddEventListener(LINK(this, DNDEventDispatcher, WindowEventListener));
}
// DNDEventDispatcher::drop
void SAL_CALL DNDEventDispatcher::drop( const DropTargetDropEvent& dtde )
throw(RuntimeException, std::exception)
{
osl::MutexGuard aImplGuard( m_aMutex );
Point location( dtde.LocationX, dtde.LocationY );
vcl::Window* pChildWindow = findTopLevelWindow(location);
// handle the case that drop is in an other vcl window than the last dragOver
if( pChildWindow != m_pCurrentWindow.get() )
{
// fire dragExit on listeners of previous window
fireDragExitEvent( m_pCurrentWindow );
fireDragEnterEvent( pChildWindow, static_cast < XDropTargetDragContext * > (this),
dtde.DropAction, location, dtde.SourceActions, m_aDataFlavorList );
}
sal_Int32 nListeners = 0;
// send drop event to the child window
nListeners = fireDropEvent( pChildWindow, dtde.Context, dtde.DropAction,
location, dtde.SourceActions, dtde.Transferable );
// reject drop if no listeners found
if( nListeners == 0 ) {
OSL_TRACE( "rejecting drop due to missing listeners." );
dtde.Context->rejectDrop();
}
// this is a drop -> no further drag overs
designate_currentwindow(nullptr);
m_aDataFlavorList.realloc( 0 );
}
// DNDEventDispatcher::dragEnter
void SAL_CALL DNDEventDispatcher::dragEnter( const DropTargetDragEnterEvent& dtdee )
throw(RuntimeException, std::exception)
{
osl::MutexGuard aImplGuard( m_aMutex );
Point location( dtdee.LocationX, dtdee.LocationY );
vcl::Window * pChildWindow = findTopLevelWindow(location);
// assume pointer write operation to be atomic
designate_currentwindow(pChildWindow);
m_aDataFlavorList = dtdee.SupportedDataFlavors;
// fire dragEnter on listeners of current window
sal_Int32 nListeners = fireDragEnterEvent( pChildWindow, dtdee.Context, dtdee.DropAction, location,
dtdee.SourceActions, dtdee.SupportedDataFlavors );
// reject drag if no listener found
if( nListeners == 0 ) {
OSL_TRACE( "rejecting drag enter due to missing listeners." );
dtdee.Context->rejectDrag();
}
}
// DNDEventDispatcher::dragExit
void SAL_CALL DNDEventDispatcher::dragExit( const DropTargetEvent& /*dte*/ )
throw(RuntimeException, std::exception)
{
osl::MutexGuard aImplGuard( m_aMutex );
fireDragExitEvent( m_pCurrentWindow );
// reset member values
designate_currentwindow(nullptr);
m_aDataFlavorList.realloc( 0 );
}
// DNDEventDispatcher::dragOver
void SAL_CALL DNDEventDispatcher::dragOver( const DropTargetDragEvent& dtde )
throw(RuntimeException, std::exception)
{
osl::MutexGuard aImplGuard( m_aMutex );
Point location( dtde.LocationX, dtde.LocationY );
sal_Int32 nListeners;
vcl::Window * pChildWindow = findTopLevelWindow(location);
if( pChildWindow != m_pCurrentWindow.get() )
{
// fire dragExit on listeners of previous window
fireDragExitEvent( m_pCurrentWindow );
// remember new window
designate_currentwindow(pChildWindow);
// fire dragEnter on listeners of current window
nListeners = fireDragEnterEvent( pChildWindow, dtde.Context, dtde.DropAction, location,
dtde.SourceActions, m_aDataFlavorList );
}
else
{
// fire dragOver on listeners of current window
nListeners = fireDragOverEvent( pChildWindow, dtde.Context, dtde.DropAction, location,
dtde.SourceActions );
}
// reject drag if no listener found
if( nListeners == 0 )
{
OSL_TRACE( "rejecting drag over due to missing listeners." );
dtde.Context->rejectDrag();
}
}
// DNDEventDispatcher::dropActionChanged
void SAL_CALL DNDEventDispatcher::dropActionChanged( const DropTargetDragEvent& dtde )
throw(RuntimeException, std::exception)
{
osl::MutexGuard aImplGuard( m_aMutex );
Point location( dtde.LocationX, dtde.LocationY );
sal_Int32 nListeners;
vcl::Window* pChildWindow = findTopLevelWindow(location);
if( pChildWindow != m_pCurrentWindow.get() )
{
// fire dragExit on listeners of previous window
fireDragExitEvent( m_pCurrentWindow );
// remember new window
designate_currentwindow(pChildWindow);
// fire dragEnter on listeners of current window
nListeners = fireDragEnterEvent( pChildWindow, dtde.Context, dtde.DropAction, location,
dtde.SourceActions, m_aDataFlavorList );
}
else
{
// fire dropActionChanged on listeners of current window
nListeners = fireDropActionChangedEvent( pChildWindow, dtde.Context, dtde.DropAction, location,
dtde.SourceActions );
}
// reject drag if no listener found
if( nListeners == 0 )
{
OSL_TRACE( "rejecting dropActionChanged due to missing listeners." );
dtde.Context->rejectDrag();
}
}
// DNDEventDispatcher::dragGestureRecognized
void SAL_CALL DNDEventDispatcher::dragGestureRecognized( const DragGestureEvent& dge )
throw(RuntimeException, std::exception)
{
osl::MutexGuard aImplGuard( m_aMutex );
Point origin( dge.DragOriginX, dge.DragOriginY );
vcl::Window* pChildWindow = findTopLevelWindow(origin);
fireDragGestureEvent( pChildWindow, dge.DragSource, dge.Event, origin, dge.DragAction );
}
// DNDEventDispatcher::disposing
void SAL_CALL DNDEventDispatcher::disposing( const EventObject& )
throw(RuntimeException, std::exception)
{
}
// DNDEventDispatcher::acceptDrag
void SAL_CALL DNDEventDispatcher::acceptDrag( sal_Int8 /*dropAction*/ ) throw(RuntimeException, std::exception)
{
}
// DNDEventDispatcher::rejectDrag
void SAL_CALL DNDEventDispatcher::rejectDrag() throw(RuntimeException, std::exception)
{
}
// DNDEventDispatcher::fireDragEnterEvent
sal_Int32 DNDEventDispatcher::fireDragEnterEvent( vcl::Window *pWindow,
const Reference< XDropTargetDragContext >& xContext, const sal_Int8 nDropAction,
const Point& rLocation, const sal_Int8 nSourceActions, const Sequence< DataFlavor >& aFlavorList
)
throw(RuntimeException)
{
sal_Int32 n = 0;
if( pWindow && pWindow->IsInputEnabled() && ! pWindow->IsInModalMode() )
{
SolarMutexClearableGuard aSolarGuard;
// set an UI lock
pWindow->IncrementLockCount();
// query DropTarget from window
Reference< XDropTarget > xDropTarget = pWindow->GetDropTarget();
if( xDropTarget.is() )
{
// retrieve relative mouse position
Point relLoc = pWindow->ImplFrameToOutput( rLocation );
aSolarGuard.clear();
n = static_cast < DNDListenerContainer * > ( xDropTarget.get() )->fireDragEnterEvent(
xContext, nDropAction, relLoc.X(), relLoc.Y(), nSourceActions, aFlavorList );
}
}
return n;
}
// DNDEventDispatcher::fireDragOverEvent
sal_Int32 DNDEventDispatcher::fireDragOverEvent( vcl::Window *pWindow,
const Reference< XDropTargetDragContext >& xContext, const sal_Int8 nDropAction,
const Point& rLocation, const sal_Int8 nSourceActions
)
throw(RuntimeException)
{
sal_Int32 n = 0;
if( pWindow && pWindow->IsInputEnabled() && ! pWindow->IsInModalMode() )
{
SolarMutexClearableGuard aSolarGuard;
// query DropTarget from window
Reference< XDropTarget > xDropTarget = pWindow->GetDropTarget();
if( xDropTarget.is() )
{
// retrieve relative mouse position
Point relLoc = pWindow->ImplFrameToOutput( rLocation );
aSolarGuard.clear();
n = static_cast < DNDListenerContainer * > ( xDropTarget.get() )->fireDragOverEvent(
xContext, nDropAction, relLoc.X(), relLoc.Y(), nSourceActions );
}
}
return n;
}
// DNDEventDispatcher::fireDragExitEvent
sal_Int32 DNDEventDispatcher::fireDragExitEvent( vcl::Window *pWindow ) throw(RuntimeException)
{
sal_Int32 n = 0;
if( pWindow && pWindow->IsInputEnabled() && ! pWindow->IsInModalMode() )
{
SolarMutexClearableGuard aGuard;
// query DropTarget from window
Reference< XDropTarget > xDropTarget = pWindow->GetDropTarget();
aGuard.clear();
if( xDropTarget.is() )
n = static_cast < DNDListenerContainer * > ( xDropTarget.get() )->fireDragExitEvent();
// release UI lock
pWindow->DecrementLockCount();
}
return n;
}
// DNDEventDispatcher::fireDropActionChangedEvent
sal_Int32 DNDEventDispatcher::fireDropActionChangedEvent( vcl::Window *pWindow,
const Reference< XDropTargetDragContext >& xContext, const sal_Int8 nDropAction,
const Point& rLocation, const sal_Int8 nSourceActions
)
throw(RuntimeException)
{
sal_Int32 n = 0;
if( pWindow && pWindow->IsInputEnabled() && ! pWindow->IsInModalMode() )
{
SolarMutexClearableGuard aGuard;
// query DropTarget from window
Reference< XDropTarget > xDropTarget = pWindow->GetDropTarget();
if( xDropTarget.is() )
{
// retrieve relative mouse position
Point relLoc = pWindow->ImplFrameToOutput( rLocation );
aGuard.clear();
n = static_cast < DNDListenerContainer * > ( xDropTarget.get() )->fireDropActionChangedEvent(
xContext, nDropAction, relLoc.X(), relLoc.Y(), nSourceActions );
}
}
return n;
}
// DNDEventDispatcher::fireDropEvent
sal_Int32 DNDEventDispatcher::fireDropEvent( vcl::Window *pWindow,
const Reference< XDropTargetDropContext >& xContext, const sal_Int8 nDropAction, const Point& rLocation,
const sal_Int8 nSourceActions, const Reference< XTransferable >& xTransferable
)
throw(RuntimeException)
{
sal_Int32 n = 0;
if( pWindow && pWindow->IsInputEnabled() && ! pWindow->IsInModalMode() )
{
SolarMutexClearableGuard aGuard;
// query DropTarget from window
Reference< XDropTarget > xDropTarget = pWindow->GetDropTarget();
// window may be destroyed in drop event handler
VclPtr<vcl::Window> xWindow = pWindow;
if( xDropTarget.is() )
{
// retrieve relative mouse position
Point relLoc = pWindow->ImplFrameToOutput( rLocation );
aGuard.clear();
n = static_cast < DNDListenerContainer * > ( xDropTarget.get() )->fireDropEvent(
xContext, nDropAction, relLoc.X(), relLoc.Y(), nSourceActions, xTransferable );
}
if ( xWindow->IsDisposed() )
{
// release UI lock
pWindow->DecrementLockCount();
}
}
return n;
}
// DNDEventDispatcher::fireDragGestureRecognized
sal_Int32 DNDEventDispatcher::fireDragGestureEvent( vcl::Window *pWindow,
const Reference< XDragSource >& xSource, const Any& event,
const Point& rOrigin, const sal_Int8 nDragAction
)
throw(css::uno::RuntimeException)
{
sal_Int32 n = 0;
if( pWindow && pWindow->IsInputEnabled() && ! pWindow->IsInModalMode() )
{
SolarMutexClearableGuard aGuard;
// query DropTarget from window
Reference< XDragGestureRecognizer > xDragGestureRecognizer = pWindow->GetDragGestureRecognizer();
if( xDragGestureRecognizer.is() )
{
// retrieve relative mouse position
Point relLoc = pWindow->ImplFrameToOutput( rOrigin );
aGuard.clear();
n = static_cast < DNDListenerContainer * > ( xDragGestureRecognizer.get() )->fireDragGestureEvent(
nDragAction, relLoc.X(), relLoc.Y(), xSource, event );
}
// release UI lock
pWindow->DecrementLockCount();
}
return n;
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */