Split the class inheritance trees for nsIDragService and nsIDragSession. Remember that, before the start of this patch series, the inheritance diagram was: nsDragService -> nsBaseDragService -> nsIDragService + nsIDragSession and the only instance was a singleton. We switched it to: nsDragService -> nsDragSession -> nsBaseDragService -> nsIDragService + nsBaseDragSession -> nsIDragSession and maintained the singleton. This allowed us to allow us to move things to the new classes without breaking behavior. We are done with that, so we can now change the inheritance to its final form: nsDragService -> nsBaseDragService -> nsIDragService nsDragSession -> nsBaseDragSession -> nsIDragSession Of course, we also need to properly construct and release the nsIDragSessions (formerly part of the singleton), so that is done here as well. That happens in nsBaseDrag[Service|Session] for parent process drags and in nsDrag[Service|Session]Proxy for content. This is all fairly straightforward, except in the case of gtk, where we need to change some callback behavior. Differential Revision: https://phabricator.services.mozilla.com/D211084
270 lines
7.0 KiB
C++
270 lines
7.0 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* 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/. */
|
|
|
|
#include "nsDragService.h"
|
|
|
|
#include "AndroidGraphics.h"
|
|
#include "AndroidWidgetUtils.h"
|
|
#include "mozilla/dom/Document.h"
|
|
#include "mozilla/java/GeckoDragAndDropWrappers.h"
|
|
#include "mozilla/PresShell.h"
|
|
#include "mozilla/ScopeExit.h"
|
|
#include "nsArrayUtils.h"
|
|
#include "nsClipboard.h"
|
|
#include "nsComponentManagerUtils.h"
|
|
#include "nsIArray.h"
|
|
#include "nsITransferable.h"
|
|
#include "nsPrimitiveHelpers.h"
|
|
#include "nsViewManager.h"
|
|
#include "nsWindow.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::widget;
|
|
|
|
StaticRefPtr<nsDragService> sDragServiceInstance;
|
|
|
|
/* static */
|
|
already_AddRefed<nsDragService> nsDragService::GetInstance() {
|
|
if (!sDragServiceInstance) {
|
|
sDragServiceInstance = new nsDragService();
|
|
ClearOnShutdown(&sDragServiceInstance);
|
|
}
|
|
|
|
RefPtr<nsDragService> service = sDragServiceInstance.get();
|
|
return service.forget();
|
|
}
|
|
|
|
already_AddRefed<nsIDragSession> nsDragService::CreateDragSession() {
|
|
RefPtr<nsDragSession> session = new nsDragSession();
|
|
return session.forget();
|
|
}
|
|
|
|
static nsWindow* GetWindow(dom::Document* aDocument) {
|
|
if (!aDocument) {
|
|
return nullptr;
|
|
}
|
|
|
|
PresShell* presShell = aDocument->GetPresShell();
|
|
if (!presShell) {
|
|
return nullptr;
|
|
}
|
|
|
|
RefPtr<nsViewManager> vm = presShell->GetViewManager();
|
|
if (!vm) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsCOMPtr<nsIWidget> widget = vm->GetRootWidget();
|
|
if (!widget) {
|
|
return nullptr;
|
|
}
|
|
|
|
RefPtr<nsWindow> window = nsWindow::From(widget);
|
|
return window.get();
|
|
}
|
|
|
|
nsresult nsDragSession::InvokeDragSessionImpl(
|
|
nsIWidget* aWidget, nsIArray* aTransferableArray,
|
|
const Maybe<CSSIntRegion>& aRegion, uint32_t aActionType) {
|
|
if (jni::GetAPIVersion() < 24) {
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
}
|
|
|
|
uint32_t count = 0;
|
|
aTransferableArray->GetLength(&count);
|
|
if (count != 1) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsCOMPtr<nsITransferable> transferable =
|
|
do_QueryElementAt(aTransferableArray, 0);
|
|
|
|
nsAutoString html;
|
|
nsAutoString text;
|
|
nsresult rv = nsClipboard::GetTextFromTransferable(transferable, text, html);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
java::GeckoDragAndDrop::SetDragData(text, html);
|
|
|
|
if (nsWindow* window = GetWindow(mSourceDocument)) {
|
|
mTransferable = transferable;
|
|
|
|
OpenDragPopup();
|
|
|
|
auto bitmap = CreateDragImage(mSourceNode, aRegion);
|
|
window->StartDragAndDrop(bitmap);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsDragSession::GetData(nsITransferable* aTransferable, uint32_t aItem) {
|
|
if (!aTransferable) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
nsTArray<nsCString> flavors;
|
|
nsresult rv = aTransferable->FlavorsTransferableCanImport(flavors);
|
|
if (NS_FAILED(rv)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
for (const auto& flavor : flavors) {
|
|
nsCOMPtr<nsISupports> data;
|
|
rv = mTransferable->GetTransferData(flavor.get(), getter_AddRefs(data));
|
|
if (NS_FAILED(rv)) {
|
|
continue;
|
|
}
|
|
rv = aTransferable->SetTransferData(flavor.get(), data);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsDragSession::GetNumDropItems(uint32_t* aNumItems) {
|
|
if (mTransferable) {
|
|
*aNumItems = 1;
|
|
return NS_OK;
|
|
}
|
|
*aNumItems = 0;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsDragSession::IsDataFlavorSupported(const char* aDataFlavor, bool* _retval) {
|
|
*_retval = false;
|
|
|
|
nsDependentCString dataFlavor(aDataFlavor);
|
|
auto logging = MakeScopeExit([&] {
|
|
MOZ_DRAGSERVICE_LOG("IsDataFlavorSupported: %s is%s found", aDataFlavor,
|
|
*_retval ? "" : " not");
|
|
});
|
|
|
|
nsTArray<nsCString> flavors;
|
|
nsresult rv = mTransferable->FlavorsTransferableCanImport(flavors);
|
|
if (NS_FAILED(rv)) {
|
|
return NS_OK;
|
|
}
|
|
|
|
for (const auto& flavor : flavors) {
|
|
if (dataFlavor.Equals(flavor)) {
|
|
*_retval = true;
|
|
return NS_OK;
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult nsDragSession::EndDragSessionImpl(bool aDoneDrag,
|
|
uint32_t aKeyModifiers) {
|
|
java::GeckoDragAndDrop::EndDragSession();
|
|
|
|
nsresult rv =
|
|
nsBaseDragSession::EndDragSessionImpl(aDoneDrag, aKeyModifiers);
|
|
mTransferable = nullptr;
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsDragSession::UpdateDragImage(nsINode* aImage, int32_t aImageX,
|
|
int32_t aImageY) {
|
|
nsBaseDragSession::UpdateDragImage(aImage, aImageX, aImageY);
|
|
auto bitmap = CreateDragImage(mSourceNode, Nothing());
|
|
|
|
if (nsWindow* window = GetWindow(mSourceDocument)) {
|
|
window->UpdateDragImage(bitmap);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
bool nsDragSession::MustUpdateDataTransfer(EventMessage aMessage) {
|
|
// Android's drag and drop API sets drop item in drop event.
|
|
// So we have to invalidate data transfer cache on drop event.
|
|
return aMessage == eDrop;
|
|
}
|
|
|
|
java::sdk::Bitmap::LocalRef nsDragSession::CreateDragImage(
|
|
nsINode* aNode, const Maybe<CSSIntRegion>& aRegion) {
|
|
LayoutDeviceIntRect dragRect;
|
|
RefPtr<SourceSurface> surface;
|
|
nsPresContext* pc;
|
|
DrawDrag(aNode, aRegion, mScreenPosition, &dragRect, &surface, &pc);
|
|
if (!surface) {
|
|
return nullptr;
|
|
}
|
|
|
|
RefPtr<DataSourceSurface> destDataSurface =
|
|
AndroidWidgetUtils::GetDataSourceSurfaceForAndroidBitmap(
|
|
surface, &dragRect, dragRect.width * 4);
|
|
if (!destDataSurface) {
|
|
return nullptr;
|
|
}
|
|
|
|
DataSourceSurface::ScopedMap destMap(destDataSurface,
|
|
DataSourceSurface::READ);
|
|
|
|
java::sdk::Bitmap::LocalRef bitmap;
|
|
auto pixels = mozilla::jni::ByteBuffer::New(
|
|
reinterpret_cast<int8_t*>(destMap.GetData()),
|
|
destMap.GetStride() * destDataSurface->GetSize().height);
|
|
bitmap = java::sdk::Bitmap::CreateBitmap(
|
|
dragRect.width, dragRect.height, java::sdk::Bitmap::Config::ARGB_8888());
|
|
bitmap->CopyPixelsFromBuffer(pixels);
|
|
return bitmap;
|
|
}
|
|
|
|
void nsDragSession::SetData(nsITransferable* aTransferable) {
|
|
mTransferable = aTransferable;
|
|
// Reset DataTransfer
|
|
mDataTransfer = nullptr;
|
|
}
|
|
|
|
void nsDragSession::SetDropData(
|
|
mozilla::java::GeckoDragAndDrop::DropData::Param aDropData) {
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (!aDropData) {
|
|
SetData(nullptr);
|
|
return;
|
|
}
|
|
|
|
nsCString mime(aDropData->MimeType()->ToCString());
|
|
|
|
if (mime.EqualsLiteral("application/x-moz-draganddrop")) {
|
|
// The drop data isn't changed.
|
|
return;
|
|
}
|
|
|
|
if (!mime.EqualsLiteral("text/plain") && !mime.EqualsLiteral("text/html")) {
|
|
// Not supported data.
|
|
SetData(nullptr);
|
|
return;
|
|
}
|
|
|
|
nsString buffer(aDropData->Text()->ToString());
|
|
nsCOMPtr<nsISupports> wrapper;
|
|
nsPrimitiveHelpers::CreatePrimitiveForData(
|
|
mime, buffer.get(), buffer.Length() * 2, getter_AddRefs(wrapper));
|
|
if (!wrapper) {
|
|
SetData(nullptr);
|
|
return;
|
|
}
|
|
nsCOMPtr<nsITransferable> transferable =
|
|
do_CreateInstance("@mozilla.org/widget/transferable;1");
|
|
transferable->Init(nullptr);
|
|
transferable->SetTransferData(mime.get(), wrapper);
|
|
SetData(transferable);
|
|
}
|