/* -*- Mode: c++; c-basic-offset: 4; tab-width: 20; indent-tabs-mode: nil; -*- * 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 "APZCCallbackHandler.h" #include "mozilla/layers/APZCCallbackHelper.h" #include "mozilla/layers/APZCTreeManager.h" #include "nsAppShell.h" #include "nsLayoutUtils.h" #include "nsPrintfCString.h" #include "nsThreadUtils.h" #include "base/message_loop.h" #include "nsWindow.h" #include "nsIInterfaceRequestorUtils.h" #include "AndroidBridge.h" #include "nsIContent.h" using mozilla::layers::APZCCallbackHelper; using mozilla::layers::APZCTreeManager; using mozilla::layers::FrameMetrics; using mozilla::layers::ScrollableLayerGuid; namespace mozilla { namespace widget { namespace android { StaticRefPtr APZCCallbackHandler::sInstance; NativePanZoomController* APZCCallbackHandler::SetNativePanZoomController(jobject obj) { NativePanZoomController* old = mNativePanZoomController; mNativePanZoomController = NativePanZoomController::Wrap(obj); return old; } void APZCCallbackHandler::NotifyDefaultPrevented(const ScrollableLayerGuid& aGuid, bool aDefaultPrevented) { if (!AndroidBridge::IsJavaUiThread()) { // The notification must reach the APZ on the Java UI thread (aka the // APZ "controller" thread) but we get it from the Gecko thread, so we // have to throw it onto the other thread. AndroidBridge::Bridge()->PostTaskToUiThread(NewRunnableMethod( this, &APZCCallbackHandler::NotifyDefaultPrevented, aGuid, aDefaultPrevented), 0); return; } MOZ_ASSERT(AndroidBridge::IsJavaUiThread()); APZCTreeManager* controller = nsWindow::GetAPZCTreeManager(); if (controller) { controller->ContentReceivedTouch(aGuid, aDefaultPrevented); } } nsIDOMWindowUtils* APZCCallbackHandler::GetDOMWindowUtils() { nsIAndroidBrowserApp* browserApp = nullptr; if (!nsAppShell::gAppShell) { return nullptr; } nsAppShell::gAppShell->GetBrowserApp(&browserApp); if (!browserApp) { return nullptr; } nsIBrowserTab* tab = nullptr; if (browserApp->GetSelectedTab(&tab) != NS_OK) { return nullptr; } nsIDOMWindow* window = nullptr; if (tab->GetWindow(&window) != NS_OK) { return nullptr; } nsCOMPtr utils = do_GetInterface(window); return utils.get(); } void APZCCallbackHandler::RequestContentRepaint(const FrameMetrics& aFrameMetrics) { MOZ_ASSERT(NS_IsMainThread()); MOZ_ASSERT(aFrameMetrics.GetScrollId() != FrameMetrics::NULL_SCROLL_ID); if (aFrameMetrics.GetIsRoot()) { nsIDOMWindowUtils* utils = GetDOMWindowUtils(); if (utils && APZCCallbackHelper::HasValidPresShellId(utils, aFrameMetrics)) { FrameMetrics metrics = aFrameMetrics; APZCCallbackHelper::UpdateRootFrame(utils, metrics); APZCCallbackHelper::UpdateCallbackTransform(aFrameMetrics, metrics); } } else { // aFrameMetrics.mIsRoot is false, so we are trying to update a subframe. // This requires special handling. nsCOMPtr content = nsLayoutUtils::FindContentFor(aFrameMetrics.GetScrollId()); if (content) { FrameMetrics newSubFrameMetrics(aFrameMetrics); APZCCallbackHelper::UpdateSubFrame(content, newSubFrameMetrics); APZCCallbackHelper::UpdateCallbackTransform(aFrameMetrics, newSubFrameMetrics); } } } void APZCCallbackHandler::AcknowledgeScrollUpdate(const FrameMetrics::ViewID& aScrollId, const uint32_t& aScrollGeneration) { APZCCallbackHelper::AcknowledgeScrollUpdate(aScrollId, aScrollGeneration); } void APZCCallbackHandler::HandleDoubleTap(const CSSPoint& aPoint, int32_t aModifiers, const mozilla::layers::ScrollableLayerGuid& aGuid) { CSSIntPoint point = RoundedToInt(aPoint); nsCString data = nsPrintfCString("{ \"x\": %d, \"y\": %d }", point.x, point.y); nsAppShell::gAppShell->PostEvent(AndroidGeckoEvent::MakeBroadcastEvent( NS_LITERAL_CSTRING("Gesture:DoubleTap"), data)); } void APZCCallbackHandler::HandleSingleTap(const CSSPoint& aPoint, int32_t aModifiers, const mozilla::layers::ScrollableLayerGuid& aGuid) { // FIXME Send the modifier data to Gecko for use in mouse events. CSSIntPoint point = RoundedToInt(aPoint); nsCString data = nsPrintfCString("{ \"x\": %d, \"y\": %d }", point.x, point.y); nsAppShell::gAppShell->PostEvent(AndroidGeckoEvent::MakeBroadcastEvent( NS_LITERAL_CSTRING("Gesture:SingleTap"), data)); } void APZCCallbackHandler::HandleLongTap(const CSSPoint& aPoint, int32_t aModifiers, const mozilla::layers::ScrollableLayerGuid& aGuid) { // TODO send content response back to APZC CSSIntPoint point = RoundedToInt(aPoint); nsCString data = nsPrintfCString("{ \"x\": %d, \"y\": %d }", point.x, point.y); nsAppShell::gAppShell->PostEvent(AndroidGeckoEvent::MakeBroadcastEvent( NS_LITERAL_CSTRING("Gesture:LongPress"), data)); } void APZCCallbackHandler::HandleLongTapUp(const CSSPoint& aPoint, int32_t aModifiers, const mozilla::layers::ScrollableLayerGuid& aGuid) { HandleSingleTap(aPoint, aModifiers, aGuid); } void APZCCallbackHandler::SendAsyncScrollDOMEvent(bool aIsRoot, const CSSRect& aContentRect, const CSSSize& aScrollableSize) { // no-op, we don't want to support this event on fennec, and we // want to get rid of this entirely. See bug 898075. } void APZCCallbackHandler::PostDelayedTask(Task* aTask, int aDelayMs) { AndroidBridge::Bridge()->PostTaskToUiThread(aTask, aDelayMs); } } // namespace android } // namespace widget } // namespace mozilla