mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-23 04:41:11 +00:00
2b82d816db
The BrowserChild should only be cleared on EndDragSession, which is sent when the user finishes or cancels dragging. This is reciprocal to it being set in StartDragSession. The change to nsDragSessionProxy::EndDragSessionImpl is non-functional -- it highlights the BrowserChild symmetry above. Differential Revision: https://phabricator.services.mozilla.com/D222389
197 lines
6.9 KiB
C++
197 lines
6.9 KiB
C++
/* -*- Mode: C++; tab-width: 4; 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 "nsDragServiceProxy.h"
|
|
#include "mozilla/dom/Document.h"
|
|
#include "mozilla/dom/BrowserChild.h"
|
|
#include "mozilla/gfx/2D.h"
|
|
#include "mozilla/net/CookieJarSettings.h"
|
|
#include "mozilla/UniquePtr.h"
|
|
#include "mozilla/Unused.h"
|
|
#include "nsContentUtils.h"
|
|
|
|
using mozilla::CSSIntRegion;
|
|
using mozilla::LayoutDeviceIntRect;
|
|
using mozilla::Maybe;
|
|
using mozilla::Nothing;
|
|
using mozilla::Some;
|
|
using mozilla::dom::BrowserChild;
|
|
using mozilla::gfx::DataSourceSurface;
|
|
using mozilla::gfx::SourceSurface;
|
|
using mozilla::gfx::SurfaceFormat;
|
|
using mozilla::ipc::Shmem;
|
|
|
|
nsDragServiceProxy::~nsDragServiceProxy() = default;
|
|
|
|
nsDragSessionProxy::~nsDragSessionProxy() = default;
|
|
|
|
already_AddRefed<nsIDragSession> nsDragServiceProxy::CreateDragSession() {
|
|
RefPtr<nsIDragSession> session = new nsDragSessionProxy();
|
|
return session.forget();
|
|
}
|
|
|
|
nsresult nsDragSessionProxy::InvokeDragSession(
|
|
nsIWidget* aWidget, nsINode* aDOMNode, nsIPrincipal* aPrincipal,
|
|
nsIContentSecurityPolicy* aCsp, nsICookieJarSettings* aCookieJarSettings,
|
|
nsIArray* aTransferableArray, uint32_t aActionType,
|
|
nsContentPolicyType aContentPolicyType) {
|
|
BrowserChild* sourceBrowser = aWidget->GetOwningBrowserChild();
|
|
NS_ENSURE_TRUE(sourceBrowser, NS_ERROR_INVALID_ARG);
|
|
[[maybe_unused]] RefPtr<nsIDragSession> sourceSession =
|
|
sourceBrowser->GetDragSession();
|
|
MOZ_ASSERT(!sourceSession);
|
|
MOZ_ALWAYS_SUCCEEDS(
|
|
sourceBrowser->GetWeakReference(getter_AddRefs(mSourceBrowser)));
|
|
sourceBrowser->SetDragSession(this);
|
|
nsresult rv = nsBaseDragSession::InvokeDragSession(
|
|
aWidget, aDOMNode, aPrincipal, aCsp, aCookieJarSettings,
|
|
aTransferableArray, aActionType, aContentPolicyType);
|
|
return rv;
|
|
}
|
|
|
|
nsresult nsDragSessionProxy::InvokeDragSessionImpl(
|
|
nsIWidget* aWidget, nsIArray* aArrayTransferables,
|
|
const Maybe<CSSIntRegion>& aRegion, uint32_t aActionType) {
|
|
NS_ENSURE_STATE(mSourceDocument->GetDocShell());
|
|
BrowserChild* child = BrowserChild::GetFrom(mSourceDocument->GetDocShell());
|
|
NS_ENSURE_STATE(child);
|
|
nsTArray<mozilla::dom::IPCTransferableData> transferables;
|
|
nsContentUtils::TransferablesToIPCTransferableDatas(
|
|
aArrayTransferables, transferables, false, nullptr);
|
|
|
|
nsCOMPtr<nsIPrincipal> principal;
|
|
if (mSourceNode) {
|
|
principal = mSourceNode->NodePrincipal();
|
|
}
|
|
|
|
nsCOMPtr<nsIContentSecurityPolicy> csp;
|
|
if (mSourceDocument) {
|
|
csp = mSourceDocument->GetCsp();
|
|
// XXX why do we need this here? Shouldn't they be set properly in
|
|
// nsBaseDragService already?
|
|
mSourceWindowContext = mSourceDocument->GetWindowContext();
|
|
mSourceTopWindowContext = mSourceWindowContext
|
|
? mSourceWindowContext->TopWindowContext()
|
|
: nullptr;
|
|
}
|
|
|
|
nsCOMPtr<nsICookieJarSettings> cookieJarSettings;
|
|
cookieJarSettings = mSourceDocument->CookieJarSettings();
|
|
mozilla::net::CookieJarSettingsArgs csArgs;
|
|
mozilla::net::CookieJarSettings::Cast(cookieJarSettings)->Serialize(csArgs);
|
|
|
|
LayoutDeviceIntRect dragRect;
|
|
if (mHasImage || mSelection) {
|
|
nsPresContext* pc;
|
|
RefPtr<SourceSurface> surface;
|
|
DrawDrag(mSourceNode, aRegion, mScreenPosition, &dragRect, &surface, &pc);
|
|
|
|
if (surface) {
|
|
RefPtr<DataSourceSurface> dataSurface = surface->GetDataSurface();
|
|
if (dataSurface) {
|
|
size_t length;
|
|
int32_t stride;
|
|
auto surfaceData =
|
|
nsContentUtils::GetSurfaceData(*dataSurface, &length, &stride);
|
|
if (surfaceData.isNothing()) {
|
|
NS_WARNING("Failed to create shared memory for drag session.");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
mozilla::Unused << child->SendInvokeDragSession(
|
|
std::move(transferables), aActionType, std::move(surfaceData),
|
|
stride, dataSurface->GetFormat(), dragRect, principal, csp, csArgs,
|
|
mSourceWindowContext, mSourceTopWindowContext);
|
|
return NS_OK;
|
|
}
|
|
}
|
|
}
|
|
|
|
mozilla::Unused << child->SendInvokeDragSession(
|
|
std::move(transferables), aActionType, Nothing(), 0,
|
|
static_cast<SurfaceFormat>(0), dragRect, principal, csp, csArgs,
|
|
mSourceWindowContext, mSourceTopWindowContext);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsIDragSession* nsDragServiceProxy::StartDragSession(
|
|
nsISupports* aWidgetProvider) {
|
|
nsIWidget* widget = GetWidgetFromWidgetProvider(aWidgetProvider);
|
|
NS_ENSURE_TRUE(widget, nullptr);
|
|
BrowserChild* targetBrowser = widget->GetOwningBrowserChild();
|
|
NS_ENSURE_TRUE(targetBrowser, nullptr);
|
|
RefPtr<nsIDragSession> session = targetBrowser->GetDragSession();
|
|
if (session) {
|
|
// session already exists on the browser
|
|
return session;
|
|
}
|
|
|
|
session = CreateDragSession();
|
|
MOZ_ASSERT(session);
|
|
static_cast<nsDragSessionProxy*>(session.get())->SetDragTarget(targetBrowser);
|
|
targetBrowser->SetDragSession(session);
|
|
return session;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsDragServiceProxy::GetCurrentSession(nsISupports* aWidgetProvider,
|
|
nsIDragSession** aSession) {
|
|
if (!aSession) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
*aSession = nullptr;
|
|
|
|
nsIWidget* widget = GetWidgetFromWidgetProvider(aWidgetProvider);
|
|
NS_ENSURE_TRUE(widget, NS_ERROR_INVALID_ARG);
|
|
BrowserChild* browser = widget->GetOwningBrowserChild();
|
|
NS_ENSURE_TRUE(browser, NS_ERROR_INVALID_ARG);
|
|
RefPtr<nsIDragSession> session = browser->GetDragSession();
|
|
|
|
if (!mSuppressLevel && session) {
|
|
session.forget(aSession);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void nsDragSessionProxy::SetDragTarget(BrowserChild* aTarget) {
|
|
if (!aTarget) {
|
|
if (mTargetBrowser) {
|
|
nsCOMPtr<BrowserChild> targetBC = do_QueryReferent(mTargetBrowser);
|
|
MOZ_ASSERT(targetBC);
|
|
if (targetBC) {
|
|
targetBC->SetDragSession(nullptr);
|
|
}
|
|
mTargetBrowser = nullptr;
|
|
}
|
|
return;
|
|
}
|
|
[[maybe_unused]] RefPtr<nsIDragSession> session = aTarget->GetDragSession();
|
|
MOZ_ASSERT(!session);
|
|
MOZ_ALWAYS_SUCCEEDS(
|
|
aTarget->GetWeakReference(getter_AddRefs(mTargetBrowser)));
|
|
}
|
|
|
|
nsresult nsDragSessionProxy::EndDragSessionImpl(bool aDoneDrag,
|
|
uint32_t aKeyModifiers) {
|
|
// End the drag session before removing it from its BrowserChild(s). This
|
|
// leaves the drag session in place while EndDragSessionImpl sends dragend.
|
|
nsresult rv = nsBaseDragSession::EndDragSessionImpl(aDoneDrag, aKeyModifiers);
|
|
|
|
if (mSourceBrowser) {
|
|
nsCOMPtr<BrowserChild> sourceBC = do_QueryReferent(mSourceBrowser);
|
|
MOZ_ASSERT(sourceBC);
|
|
[[maybe_unused]] RefPtr<nsIDragSession> session =
|
|
sourceBC->GetDragSession();
|
|
MOZ_ASSERT(session == this);
|
|
sourceBC->SetDragSession(nullptr);
|
|
mSourceBrowser = nullptr;
|
|
}
|
|
|
|
SetDragTarget(nullptr);
|
|
return rv;
|
|
}
|