| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright (C) 2011 Apple Inc. All rights reserved. | 2 * Copyright (C) 2011 Apple Inc. All rights reserved. |
| 3 * | 3 * |
| 4 * Redistribution and use in source and binary forms, with or without | 4 * Redistribution and use in source and binary forms, with or without |
| 5 * modification, are permitted provided that the following conditions | 5 * modification, are permitted provided that the following conditions |
| 6 * are met: | 6 * are met: |
| 7 * 1. Redistributions of source code must retain the above copyright | 7 * 1. Redistributions of source code must retain the above copyright |
| 8 * notice, this list of conditions and the following disclaimer. | 8 * notice, this list of conditions and the following disclaimer. |
| 9 * 2. Redistributions in binary form must reproduce the above copyright | 9 * 2. Redistributions in binary form must reproduce the above copyright |
| 10 * notice, this list of conditions and the following disclaimer in the | 10 * notice, this list of conditions and the following disclaimer in the |
| 11 * documentation and/or other materials provided with the distribution. | 11 * documentation and/or other materials provided with the distribution. |
| 12 * | 12 * |
| 13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' | 13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' |
| 14 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, | 14 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, |
| 15 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | 15 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| 16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS | 16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS |
| 17 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | 17 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| 18 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | 18 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| 19 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | 19 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| 20 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | 20 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| 21 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | 21 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| 22 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF | 22 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF |
| 23 * THE POSSIBILITY OF SUCH DAMAGE. | 23 * THE POSSIBILITY OF SUCH DAMAGE. |
| 24 */ | 24 */ |
| 25 | 25 |
| 26 #include "config.h" | 26 #include "config.h" |
| 27 | 27 |
| 28 #include "core/page/scrolling/ScrollingCoordinator.h" | 28 #include "core/page/scrolling/ScrollingCoordinator.h" |
| 29 | 29 |
| 30 #include "RuntimeEnabledFeatures.h" |
| 30 #include "core/dom/Document.h" | 31 #include "core/dom/Document.h" |
| 32 #include "core/dom/Node.h" |
| 33 #include "core/html/HTMLElement.h" |
| 31 #include "core/page/Frame.h" | 34 #include "core/page/Frame.h" |
| 32 #include "core/page/FrameView.h" | 35 #include "core/page/FrameView.h" |
| 33 #include "core/page/Page.h" | 36 #include "core/page/Page.h" |
| 34 #include "core/platform/PlatformWheelEvent.h" | 37 #include "core/platform/PlatformWheelEvent.h" |
| 35 #include "core/platform/ScrollAnimator.h" | 38 #include "core/platform/ScrollAnimator.h" |
| 36 #include "core/platform/ScrollbarThemeComposite.h" | 39 #include "core/platform/ScrollbarThemeComposite.h" |
| 40 #include "core/platform/chromium/TraceEvent.h" |
| 37 #include "core/platform/chromium/support/WebScrollbarImpl.h" | 41 #include "core/platform/chromium/support/WebScrollbarImpl.h" |
| 38 #include "core/platform/chromium/support/WebScrollbarThemeGeometryNative.h" | 42 #include "core/platform/chromium/support/WebScrollbarThemeGeometryNative.h" |
| 39 #include "core/platform/graphics/GraphicsLayer.h" | 43 #include "core/platform/graphics/GraphicsLayer.h" |
| 40 #include "core/platform/graphics/IntRect.h" | 44 #include "core/platform/graphics/IntRect.h" |
| 41 #include "core/platform/graphics/Region.h" | 45 #include "core/platform/graphics/Region.h" |
| 46 #include "core/platform/graphics/transforms/TransformState.h" |
| 42 #include "core/plugins/PluginView.h" | 47 #include "core/plugins/PluginView.h" |
| 43 #include "core/rendering/RenderLayerBacking.h" | 48 #include "core/rendering/RenderLayerBacking.h" |
| 44 #include "core/rendering/RenderLayerCompositor.h" | 49 #include "core/rendering/RenderLayerCompositor.h" |
| 45 #include "core/rendering/RenderView.h" | 50 #include "core/rendering/RenderView.h" |
| 46 #include "public/platform/Platform.h" | 51 #include "public/platform/Platform.h" |
| 47 #include "public/platform/WebCompositorSupport.h" | 52 #include "public/platform/WebCompositorSupport.h" |
| 48 #include "public/platform/WebLayerPositionConstraint.h" | 53 #include "public/platform/WebLayerPositionConstraint.h" |
| 49 #include "public/platform/WebScrollbarLayer.h" | 54 #include "public/platform/WebScrollbarLayer.h" |
| 50 #include "public/platform/WebScrollbarThemeGeometry.h" | 55 #include "public/platform/WebScrollbarThemeGeometry.h" |
| 51 #include "public/platform/WebScrollbarThemePainter.h" | 56 #include "public/platform/WebScrollbarThemePainter.h" |
| (...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 84 ScrollingCoordinator::~ScrollingCoordinator() | 89 ScrollingCoordinator::~ScrollingCoordinator() |
| 85 { | 90 { |
| 86 ASSERT(!m_page); | 91 ASSERT(!m_page); |
| 87 for (ScrollbarMap::iterator it = m_horizontalScrollbars.begin(); it != m_hor
izontalScrollbars.end(); ++it) | 92 for (ScrollbarMap::iterator it = m_horizontalScrollbars.begin(); it != m_hor
izontalScrollbars.end(); ++it) |
| 88 GraphicsLayer::unregisterContentsLayer(it->value->layer()); | 93 GraphicsLayer::unregisterContentsLayer(it->value->layer()); |
| 89 for (ScrollbarMap::iterator it = m_verticalScrollbars.begin(); it != m_verti
calScrollbars.end(); ++it) | 94 for (ScrollbarMap::iterator it = m_verticalScrollbars.begin(); it != m_verti
calScrollbars.end(); ++it) |
| 90 GraphicsLayer::unregisterContentsLayer(it->value->layer()); | 95 GraphicsLayer::unregisterContentsLayer(it->value->layer()); |
| 91 | 96 |
| 92 } | 97 } |
| 93 | 98 |
| 99 bool ScrollingCoordinator::touchHitTestingEnabled() const |
| 100 { |
| 101 RenderView* contentRenderer = m_page->mainFrame()->contentRenderer(); |
| 102 return RuntimeEnabledFeatures::touchEnabled() && contentRenderer && contentR
enderer->usesCompositing(); |
| 103 } |
| 104 |
| 94 void ScrollingCoordinator::setShouldHandleScrollGestureOnMainThreadRegion(const
Region& region) | 105 void ScrollingCoordinator::setShouldHandleScrollGestureOnMainThreadRegion(const
Region& region) |
| 95 { | 106 { |
| 96 if (WebLayer* scrollLayer = scrollingWebLayerForScrollableArea(m_page->mainF
rame()->view())) { | 107 if (WebLayer* scrollLayer = scrollingWebLayerForScrollableArea(m_page->mainF
rame()->view())) { |
| 97 Vector<IntRect> rects = region.rects(); | 108 Vector<IntRect> rects = region.rects(); |
| 98 WebVector<WebRect> webRects(rects.size()); | 109 WebVector<WebRect> webRects(rects.size()); |
| 99 for (size_t i = 0; i < rects.size(); ++i) | 110 for (size_t i = 0; i < rects.size(); ++i) |
| 100 webRects[i] = rects[i]; | 111 webRects[i] = rects[i]; |
| 101 scrollLayer->setNonFastScrollableRegion(webRects); | 112 scrollLayer->setNonFastScrollableRegion(webRects); |
| 102 } | 113 } |
| 103 } | 114 } |
| 104 | 115 |
| 105 void ScrollingCoordinator::frameViewLayoutUpdated(FrameView* frameView) | 116 void ScrollingCoordinator::frameViewLayoutUpdated(FrameView* frameView) |
| 106 { | 117 { |
| 107 ASSERT(m_page); | 118 if (!touchHitTestingEnabled()) |
| 119 return; |
| 120 |
| 121 TRACE_EVENT0("input", "ScrollingCoordinator::frameViewLayoutUpdated"); |
| 108 | 122 |
| 109 // Compute the region of the page that we can't handle scroll gestures on im
pl thread: | 123 // Compute the region of the page that we can't handle scroll gestures on im
pl thread: |
| 110 // This currently includes: | 124 // This currently includes: |
| 111 // 1. All scrollable areas, such as subframes, overflow divs and list boxes,
whose composited | 125 // 1. All scrollable areas, such as subframes, overflow divs and list boxes,
whose composited |
| 112 // scrolling are not enabled. We need to do this even if the frame view whos
e layout was updated | 126 // scrolling are not enabled. We need to do this even if the frame view whos
e layout was updated |
| 113 // is not the main frame. | 127 // is not the main frame. |
| 114 // 2. Resize control areas, e.g. the small rect at the right bottom of div/t
extarea/iframe when | 128 // 2. Resize control areas, e.g. the small rect at the right bottom of div/t
extarea/iframe when |
| 115 // CSS property "resize" is enabled. | 129 // CSS property "resize" is enabled. |
| 116 // 3. Plugin areas. | 130 // 3. Plugin areas. |
| 117 Region shouldHandleScrollGestureOnMainThreadRegion = computeShouldHandleScro
llGestureOnMainThreadRegion(m_page->mainFrame(), IntPoint()); | 131 Region shouldHandleScrollGestureOnMainThreadRegion = computeShouldHandleScro
llGestureOnMainThreadRegion(m_page->mainFrame(), IntPoint()); |
| 118 setShouldHandleScrollGestureOnMainThreadRegion(shouldHandleScrollGestureOnMa
inThreadRegion); | 132 setShouldHandleScrollGestureOnMainThreadRegion(shouldHandleScrollGestureOnMa
inThreadRegion); |
| 119 Vector<IntRect> touchEventTargetRects; | 133 LayerHitTestRects touchEventTargetRects; |
| 120 computeAbsoluteTouchEventTargetRects(m_page->mainFrame()->document(), touchE
ventTargetRects); | 134 computeTouchEventTargetRects(touchEventTargetRects); |
| 121 setTouchEventTargetRects(touchEventTargetRects); | 135 setTouchEventTargetRects(touchEventTargetRects); |
| 122 if (WebLayer* scrollLayer = scrollingWebLayerForScrollableArea(frameView)) | 136 if (WebLayer* scrollLayer = scrollingWebLayerForScrollableArea(frameView)) |
| 123 scrollLayer->setBounds(frameView->contentsSize()); | 137 scrollLayer->setBounds(frameView->contentsSize()); |
| 124 } | 138 } |
| 125 | 139 |
| 126 void ScrollingCoordinator::setLayerIsContainerForFixedPositionLayers(GraphicsLay
er* layer, bool enable) | 140 void ScrollingCoordinator::setLayerIsContainerForFixedPositionLayers(GraphicsLay
er* layer, bool enable) |
| 127 { | 141 { |
| 128 if (WebLayer* scrollableLayer = scrollingWebLayerForGraphicsLayer(layer)) | 142 if (WebLayer* scrollableLayer = scrollingWebLayerForGraphicsLayer(layer)) |
| 129 scrollableLayer->setIsContainerForFixedPositionLayers(enable); | 143 scrollableLayer->setIsContainerForFixedPositionLayers(enable); |
| 130 } | 144 } |
| (...skipping 158 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 289 } | 303 } |
| 290 if (WebScrollbarLayer* scrollbarLayer = getWebScrollbarLayer(scrollableArea,
VerticalScrollbar)) { | 304 if (WebScrollbarLayer* scrollbarLayer = getWebScrollbarLayer(scrollableArea,
VerticalScrollbar)) { |
| 291 GraphicsLayer* verticalScrollbarLayer = verticalScrollbarLayerForScrolla
bleArea(scrollableArea); | 305 GraphicsLayer* verticalScrollbarLayer = verticalScrollbarLayerForScrolla
bleArea(scrollableArea); |
| 292 if (verticalScrollbarLayer) | 306 if (verticalScrollbarLayer) |
| 293 setupScrollbarLayer(verticalScrollbarLayer, scrollbarLayer, webLayer
); | 307 setupScrollbarLayer(verticalScrollbarLayer, scrollbarLayer, webLayer
); |
| 294 } | 308 } |
| 295 | 309 |
| 296 return !!webLayer; | 310 return !!webLayer; |
| 297 } | 311 } |
| 298 | 312 |
| 299 void ScrollingCoordinator::setTouchEventTargetRects(const Vector<IntRect>& absol
uteHitTestRects) | 313 static void convertLayerRectsToEnclosingCompositedLayer(const LayerHitTestRects&
layerRects, LayerHitTestRects& compositorRects) |
| 300 { | 314 { |
| 301 if (WebLayer* scrollLayer = scrollingWebLayerForScrollableArea(m_page->mainF
rame()->view())) { | 315 // We have a set of rects per RenderLayer, we need to map them to their boun
ding boxes in their |
| 302 WebVector<WebRect> webRects(absoluteHitTestRects.size()); | 316 // enclosing composited layer. |
| 303 for (size_t i = 0; i < absoluteHitTestRects.size(); ++i) | 317 for (LayerHitTestRects::const_iterator layerIter = layerRects.begin(); layer
Iter != layerRects.end(); ++layerIter) { |
| 304 webRects[i] = absoluteHitTestRects[i]; | 318 // Find the enclosing composited layer when it's in another document (fo
r non-composited iframes). |
| 305 scrollLayer->setTouchEventHandlerRegion(webRects); | 319 RenderLayer* compositedLayer = 0; |
| 320 for (const RenderLayer* layer = layerIter->key; !compositedLayer;) { |
| 321 compositedLayer = layer->enclosingCompositingLayerForRepaint(); |
| 322 if (!compositedLayer) { |
| 323 RenderObject* owner = layer->renderer()->frame()->ownerRenderer(
); |
| 324 if (!owner) |
| 325 break; |
| 326 layer = owner->enclosingLayer(); |
| 327 } |
| 328 } |
| 329 if (!compositedLayer) { |
| 330 // Since this machinery is used only when accelerated compositing is
enabled, we expect |
| 331 // that every layer should have an enclosing composited layer. |
| 332 ASSERT_NOT_REACHED(); |
| 333 continue; |
| 334 } |
| 335 |
| 336 LayerHitTestRects::iterator compIter = compositorRects.find(compositedLa
yer); |
| 337 if (compIter == compositorRects.end()) |
| 338 compIter = compositorRects.add(compositedLayer, Vector<LayoutRect>()
).iterator; |
| 339 |
| 340 // Transform each rect to the co-ordinate space of it's enclosing compos
ited layer. |
| 341 // Ideally we'd compute a transformation matrix once and re-use it for e
ach rect, but |
| 342 // there doesn't appear to be any easy way to do it (mapLocalToContainer
will flatten |
| 343 // the TransformState, so we can't use setQuad/mappedQuad over and over
again). Perhaps |
| 344 // RenderGeometryMap? |
| 345 for (size_t i = 0; i < layerIter->value.size(); ++i) { |
| 346 FloatQuad localQuad(layerIter->value[i]); |
| 347 TransformState transformState(TransformState::ApplyTransformDirectio
n, localQuad); |
| 348 MapCoordinatesFlags flags = ApplyContainerFlip | UseTransforms | Tra
verseDocumentBoundaries; |
| 349 layerIter->key->renderer()->mapLocalToContainer(compositedLayer->ren
derer(), transformState, flags); |
| 350 transformState.flatten(); |
| 351 LayoutRect compositorRect = LayoutRect(transformState.lastPlanarQuad
().boundingBox()); |
| 352 compIter->value.append(compositorRect); |
| 353 } |
| 354 } |
| 355 } |
| 356 |
| 357 void ScrollingCoordinator::setTouchEventTargetRects(const LayerHitTestRects& lay
erRects) |
| 358 { |
| 359 TRACE_EVENT0("input", "ScrollingCoordinator::setTouchEventTargetRects"); |
| 360 |
| 361 LayerHitTestRects compositorRects; |
| 362 convertLayerRectsToEnclosingCompositedLayer(layerRects, compositorRects); |
| 363 |
| 364 // Inform any observers (i.e. for testing) of these new rects. |
| 365 HashSet<TouchEventTargetRectsObserver*>::iterator stop = m_touchEventTargetR
ectsObservers.end(); |
| 366 for (HashSet<TouchEventTargetRectsObserver*>::iterator it = m_touchEventTarg
etRectsObservers.begin(); it != stop; ++it) |
| 367 (*it)->touchEventTargetRectsChanged(compositorRects); |
| 368 |
| 369 // Note that ideally we'd clear the touch event handler region on all layers
first, |
| 370 // in case there are others that no longer have any handlers. But it's unlik
ely to |
| 371 // matter much in practice (just makes us more conservative). |
| 372 for (LayerHitTestRects::const_iterator iter = compositorRects.begin(); iter
!= compositorRects.end(); ++iter) { |
| 373 WebVector<WebRect> webRects(iter->value.size()); |
| 374 for (size_t i = 0; i < iter->value.size(); ++i) |
| 375 webRects[i] = enclosingIntRect(iter->value[i]); |
| 376 RenderLayerBacking* backing = iter->key->backing(); |
| 377 // If the layer is using composited scrolling, then it's the contents th
at these |
| 378 // rects apply to. |
| 379 GraphicsLayer* graphicsLayer = backing->scrollingContentsLayer(); |
| 380 if (!graphicsLayer) |
| 381 graphicsLayer = backing->graphicsLayer(); |
| 382 graphicsLayer->platformLayer()->setTouchEventHandlerRegion(webRects); |
| 306 } | 383 } |
| 307 } | 384 } |
| 308 | 385 |
| 309 void ScrollingCoordinator::touchEventTargetRectsDidChange(const Document*) | 386 void ScrollingCoordinator::touchEventTargetRectsDidChange(const Document*) |
| 310 { | 387 { |
| 311 // The rects are always evaluated and used in the main frame coordinates. | 388 if (!touchHitTestingEnabled()) |
| 312 FrameView* frameView = m_page->mainFrame()->view(); | 389 return; |
| 313 Document* document = m_page->mainFrame()->document(); | |
| 314 | 390 |
| 315 // Wait until after layout to update. | 391 // Wait until after layout to update. |
| 316 if (frameView->needsLayout() || !document) | 392 if (m_page->mainFrame()->view()->needsLayout()) |
| 317 return; | 393 return; |
| 318 | 394 |
| 319 Vector<IntRect> touchEventTargetRects; | 395 TRACE_EVENT0("input", "ScrollingCoordinator::touchEventTargetRectsDidChange"
); |
| 320 computeAbsoluteTouchEventTargetRects(document, touchEventTargetRects); | 396 |
| 397 LayerHitTestRects touchEventTargetRects; |
| 398 computeTouchEventTargetRects(touchEventTargetRects); |
| 321 setTouchEventTargetRects(touchEventTargetRects); | 399 setTouchEventTargetRects(touchEventTargetRects); |
| 322 } | 400 } |
| 323 | 401 |
| 324 void ScrollingCoordinator::setWheelEventHandlerCount(unsigned count) | 402 void ScrollingCoordinator::setWheelEventHandlerCount(unsigned count) |
| 325 { | 403 { |
| 326 if (WebLayer* scrollLayer = scrollingWebLayerForScrollableArea(m_page->mainF
rame()->view())) | 404 if (WebLayer* scrollLayer = scrollingWebLayerForScrollableArea(m_page->mainF
rame()->view())) |
| 327 scrollLayer->setHaveWheelEventHandlers(count > 0); | 405 scrollLayer->setHaveWheelEventHandlers(count > 0); |
| 328 } | 406 } |
| 329 | 407 |
| 330 void ScrollingCoordinator::recomputeWheelEventHandlerCountForFrameView(FrameView
* frameView) | 408 void ScrollingCoordinator::recomputeWheelEventHandlerCountForFrameView(FrameView
* frameView) |
| (...skipping 25 matching lines...) Expand all Loading... |
| 356 | 434 |
| 357 // We currently only support composited mode. | 435 // We currently only support composited mode. |
| 358 RenderView* renderView = m_page->mainFrame()->contentRenderer(); | 436 RenderView* renderView = m_page->mainFrame()->contentRenderer(); |
| 359 if (!renderView) | 437 if (!renderView) |
| 360 return false; | 438 return false; |
| 361 return renderView->usesCompositing(); | 439 return renderView->usesCompositing(); |
| 362 } | 440 } |
| 363 | 441 |
| 364 Region ScrollingCoordinator::computeShouldHandleScrollGestureOnMainThreadRegion(
const Frame* frame, const IntPoint& frameLocation) const | 442 Region ScrollingCoordinator::computeShouldHandleScrollGestureOnMainThreadRegion(
const Frame* frame, const IntPoint& frameLocation) const |
| 365 { | 443 { |
| 444 TRACE_EVENT0("input", "ScrollingCoordinator::computeShouldHandleScrollGestur
eOnMainThreadRegion"); |
| 366 Region shouldHandleScrollGestureOnMainThreadRegion; | 445 Region shouldHandleScrollGestureOnMainThreadRegion; |
| 367 FrameView* frameView = frame->view(); | 446 FrameView* frameView = frame->view(); |
| 368 if (!frameView) | 447 if (!frameView) |
| 369 return shouldHandleScrollGestureOnMainThreadRegion; | 448 return shouldHandleScrollGestureOnMainThreadRegion; |
| 370 | 449 |
| 371 IntPoint offset = frameLocation; | 450 IntPoint offset = frameLocation; |
| 372 offset.moveBy(frameView->frameRect().location()); | 451 offset.moveBy(frameView->frameRect().location()); |
| 373 | 452 |
| 374 if (const FrameView::ScrollableAreaSet* scrollableAreas = frameView->scrolla
bleAreas()) { | 453 if (const FrameView::ScrollableAreaSet* scrollableAreas = frameView->scrolla
bleAreas()) { |
| 375 for (FrameView::ScrollableAreaSet::const_iterator it = scrollableAreas->
begin(), end = scrollableAreas->end(); it != end; ++it) { | 454 for (FrameView::ScrollableAreaSet::const_iterator it = scrollableAreas->
begin(), end = scrollableAreas->end(); it != end; ++it) { |
| (...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 408 } | 487 } |
| 409 } | 488 } |
| 410 | 489 |
| 411 FrameTree* tree = frame->tree(); | 490 FrameTree* tree = frame->tree(); |
| 412 for (Frame* subFrame = tree->firstChild(); subFrame; subFrame = subFrame->tr
ee()->nextSibling()) | 491 for (Frame* subFrame = tree->firstChild(); subFrame; subFrame = subFrame->tr
ee()->nextSibling()) |
| 413 shouldHandleScrollGestureOnMainThreadRegion.unite(computeShouldHandleScr
ollGestureOnMainThreadRegion(subFrame, offset)); | 492 shouldHandleScrollGestureOnMainThreadRegion.unite(computeShouldHandleScr
ollGestureOnMainThreadRegion(subFrame, offset)); |
| 414 | 493 |
| 415 return shouldHandleScrollGestureOnMainThreadRegion; | 494 return shouldHandleScrollGestureOnMainThreadRegion; |
| 416 } | 495 } |
| 417 | 496 |
| 418 static void accumulateRendererTouchEventTargetRects(Vector<IntRect>& rects, cons
t RenderObject* renderer, const IntRect& parentRect = IntRect()) | 497 void ScrollingCoordinator::addTouchEventTargetRectsObserver(TouchEventTargetRect
sObserver* observer) |
| 419 { | 498 { |
| 420 IntRect adjustedParentRect = parentRect; | 499 m_touchEventTargetRectsObservers.add(observer); |
| 421 if (parentRect.isEmpty() || renderer->isFloating() || renderer->isPositioned
() || renderer->hasTransform()) { | |
| 422 // FIXME: This method is O(N^2) as it walks the tree to the root for eve
ry renderer. RenderGeometryMap would fix this. | |
| 423 IntRect r = enclosingIntRect(renderer->clippedOverflowRectForRepaint(0))
; | |
| 424 if (!r.isEmpty()) { | |
| 425 // Convert to the top-level view's coordinates. | |
| 426 ASSERT(renderer->document()->view()); | |
| 427 r = renderer->document()->view()->convertToRootView(r); | |
| 428 | |
| 429 if (!parentRect.contains(r)) { | |
| 430 rects.append(r); | |
| 431 adjustedParentRect = r; | |
| 432 } | |
| 433 } | |
| 434 } | |
| 435 | |
| 436 for (RenderObject* child = renderer->firstChild(); child; child = child->nex
tSibling()) | |
| 437 accumulateRendererTouchEventTargetRects(rects, child, adjustedParentRect
); | |
| 438 } | 500 } |
| 439 | 501 |
| 440 static void accumulateDocumentEventTargetRects(Vector<IntRect>& rects, const Doc
ument* document) | 502 void ScrollingCoordinator::removeTouchEventTargetRectsObserver(TouchEventTargetR
ectsObserver* observer) |
| 503 { |
| 504 m_touchEventTargetRectsObservers.remove(observer); |
| 505 } |
| 506 |
| 507 static void accumulateDocumentTouchEventTargetRects(LayerHitTestRects& rects, co
nst Document* document) |
| 441 { | 508 { |
| 442 ASSERT(document); | 509 ASSERT(document); |
| 443 if (!document->touchEventTargets()) | 510 if (!document->touchEventTargets()) |
| 444 return; | 511 return; |
| 445 | 512 |
| 446 const TouchEventTargetSet* targets = document->touchEventTargets(); | 513 const TouchEventTargetSet* targets = document->touchEventTargets(); |
| 514 |
| 515 // If there's a handler on the document, html or body element (fairly common
in practice), |
| 516 // then we can quickly mark the entire document and skip looking at any othe
r handlers. |
| 517 // Note that technically a handler on the body doesn't cover the whole docum
ent, but it's |
| 518 // reasonable to be conservative and report the whole document anyway. |
| 447 for (TouchEventTargetSet::const_iterator iter = targets->begin(); iter != ta
rgets->end(); ++iter) { | 519 for (TouchEventTargetSet::const_iterator iter = targets->begin(); iter != ta
rgets->end(); ++iter) { |
| 448 const Node* touchTarget = iter->key; | 520 Node* target = iter->key; |
| 449 if (!touchTarget->inDocument()) | 521 if (target == document || target == document->documentElement() || targe
t == document->body()) { |
| 450 continue; | 522 if (RenderObject* renderer = document->renderer()) { |
| 451 | 523 renderer->computeLayerHitTestRects(rects); |
| 452 if (touchTarget == document) { | |
| 453 if (RenderView* view = document->renderView()) { | |
| 454 IntRect r; | |
| 455 if (touchTarget == document->topDocument()) | |
| 456 r = view->documentRect(); | |
| 457 else | |
| 458 r = enclosingIntRect(view->clippedOverflowRectForRepaint(0))
; | |
| 459 | |
| 460 if (!r.isEmpty()) { | |
| 461 ASSERT(view->document()->view()); | |
| 462 r = view->document()->view()->convertToRootView(r); | |
| 463 rects.append(r); | |
| 464 } | |
| 465 } | 524 } |
| 466 return; | 525 return; |
| 467 } | 526 } |
| 527 } |
| 468 | 528 |
| 469 if (touchTarget->isDocumentNode() && touchTarget != document) { | 529 for (TouchEventTargetSet::const_iterator iter = targets->begin(); iter != ta
rgets->end(); ++iter) { |
| 470 accumulateDocumentEventTargetRects(rects, toDocument(touchTarget)); | 530 const Node* target = iter->key; |
| 531 if (!target->inDocument()) |
| 471 continue; | 532 continue; |
| 533 |
| 534 if (target->isDocumentNode()) { |
| 535 ASSERT(target != document); |
| 536 accumulateDocumentTouchEventTargetRects(rects, toDocument(target)); |
| 537 } else if (RenderObject* renderer = target->renderer()) { |
| 538 renderer->computeLayerHitTestRects(rects); |
| 472 } | 539 } |
| 540 } |
| 473 | 541 |
| 474 if (RenderObject* renderer = touchTarget->renderer()) | |
| 475 accumulateRendererTouchEventTargetRects(rects, renderer); | |
| 476 } | |
| 477 } | 542 } |
| 478 | 543 |
| 479 void ScrollingCoordinator::computeAbsoluteTouchEventTargetRects(const Document*
document, Vector<IntRect>& rects) | 544 void ScrollingCoordinator::computeTouchEventTargetRects(LayerHitTestRects& rects
) |
| 480 { | 545 { |
| 481 ASSERT(document); | 546 TRACE_EVENT0("input", "ScrollingCoordinator::computeTouchEventTargetRects"); |
| 482 if (!document->view()) | 547 ASSERT(touchHitTestingEnabled()); |
| 548 |
| 549 Document* document = m_page->mainFrame()->document(); |
| 550 if (!document || !document->view()) |
| 483 return; | 551 return; |
| 484 | 552 |
| 485 // FIXME: These rects won't be properly updated if the renderers are in a su
b-tree that scrolls. | 553 accumulateDocumentTouchEventTargetRects(rects, document); |
| 486 accumulateDocumentEventTargetRects(rects, document); | |
| 487 } | 554 } |
| 488 | 555 |
| 489 unsigned ScrollingCoordinator::computeCurrentWheelEventHandlerCount() | 556 unsigned ScrollingCoordinator::computeCurrentWheelEventHandlerCount() |
| 490 { | 557 { |
| 491 unsigned wheelEventHandlerCount = 0; | 558 unsigned wheelEventHandlerCount = 0; |
| 492 | 559 |
| 493 for (Frame* frame = m_page->mainFrame(); frame; frame = frame->tree()->trave
rseNext()) { | 560 for (Frame* frame = m_page->mainFrame(); frame; frame = frame->tree()->trave
rseNext()) { |
| 494 if (frame->document()) | 561 if (frame->document()) |
| 495 wheelEventHandlerCount += frame->document()->wheelEventHandlerCount(
); | 562 wheelEventHandlerCount += frame->document()->wheelEventHandlerCount(
); |
| 496 } | 563 } |
| (...skipping 151 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 648 stringBuilder.resize(stringBuilder.length() - 2); | 715 stringBuilder.resize(stringBuilder.length() - 2); |
| 649 return stringBuilder.toString(); | 716 return stringBuilder.toString(); |
| 650 } | 717 } |
| 651 | 718 |
| 652 String ScrollingCoordinator::mainThreadScrollingReasonsAsText() const | 719 String ScrollingCoordinator::mainThreadScrollingReasonsAsText() const |
| 653 { | 720 { |
| 654 return mainThreadScrollingReasonsAsText(mainThreadScrollingReasons()); | 721 return mainThreadScrollingReasonsAsText(mainThreadScrollingReasons()); |
| 655 } | 722 } |
| 656 | 723 |
| 657 } // namespace WebCore | 724 } // namespace WebCore |
| OLD | NEW |