| /* |
| * Copyright (C) 2006, 2007, 2008, 2014 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR |
| * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
| * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
| * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "config.h" |
| #include "ScrollView.h" |
| |
| #include "GraphicsContext.h" |
| #include "GraphicsLayer.h" |
| #include "HostWindow.h" |
| #include "PlatformMouseEvent.h" |
| #include "PlatformWheelEvent.h" |
| #include "ScrollAnimator.h" |
| #include "Scrollbar.h" |
| #include "ScrollbarTheme.h" |
| #include <wtf/StdLibExtras.h> |
| |
| namespace WebCore { |
| |
| ScrollView::ScrollView() |
| : m_horizontalScrollbarMode(ScrollbarAuto) |
| , m_verticalScrollbarMode(ScrollbarAuto) |
| , m_horizontalScrollbarLock(false) |
| , m_verticalScrollbarLock(false) |
| , m_prohibitsScrolling(false) |
| , m_canBlitOnScroll(true) |
| , m_scrollbarsAvoidingResizer(0) |
| , m_scrollbarsSuppressed(false) |
| , m_inUpdateScrollbars(false) |
| , m_updateScrollbarsPass(0) |
| , m_drawPanScrollIcon(false) |
| , m_useFixedLayout(false) |
| , m_paintsEntireContents(false) |
| , m_clipsRepaints(true) |
| , m_delegatesScrolling(false) |
| { |
| } |
| |
| ScrollView::~ScrollView() |
| { |
| } |
| |
| void ScrollView::addChild(PassRefPtr<Widget> prpChild) |
| { |
| Widget* child = prpChild.get(); |
| ASSERT(child != this && !child->parent()); |
| child->setParent(this); |
| m_children.add(prpChild); |
| if (child->platformWidget()) |
| platformAddChild(child); |
| } |
| |
| void ScrollView::removeChild(Widget* child) |
| { |
| ASSERT(child->parent() == this); |
| child->setParent(0); |
| m_children.remove(child); |
| if (child->platformWidget()) |
| platformRemoveChild(child); |
| } |
| |
| bool ScrollView::setHasHorizontalScrollbar(bool hasBar, bool* contentSizeAffected) |
| { |
| ASSERT(!hasBar || !avoidScrollbarCreation()); |
| if (hasBar && !m_horizontalScrollbar) { |
| m_horizontalScrollbar = createScrollbar(HorizontalScrollbar); |
| addChild(m_horizontalScrollbar.get()); |
| didAddScrollbar(m_horizontalScrollbar.get(), HorizontalScrollbar); |
| m_horizontalScrollbar->styleChanged(); |
| if (contentSizeAffected) |
| *contentSizeAffected = !m_horizontalScrollbar->isOverlayScrollbar(); |
| return true; |
| } |
| |
| if (!hasBar && m_horizontalScrollbar) { |
| bool wasOverlayScrollbar = m_horizontalScrollbar->isOverlayScrollbar(); |
| willRemoveScrollbar(m_horizontalScrollbar.get(), HorizontalScrollbar); |
| removeChild(m_horizontalScrollbar.get()); |
| m_horizontalScrollbar = 0; |
| if (contentSizeAffected) |
| *contentSizeAffected = !wasOverlayScrollbar; |
| return true; |
| } |
| |
| return false; |
| } |
| |
| bool ScrollView::setHasVerticalScrollbar(bool hasBar, bool* contentSizeAffected) |
| { |
| ASSERT(!hasBar || !avoidScrollbarCreation()); |
| if (hasBar && !m_verticalScrollbar) { |
| m_verticalScrollbar = createScrollbar(VerticalScrollbar); |
| addChild(m_verticalScrollbar.get()); |
| didAddScrollbar(m_verticalScrollbar.get(), VerticalScrollbar); |
| m_verticalScrollbar->styleChanged(); |
| if (contentSizeAffected) |
| *contentSizeAffected = !m_verticalScrollbar->isOverlayScrollbar(); |
| return true; |
| } |
| |
| if (!hasBar && m_verticalScrollbar) { |
| bool wasOverlayScrollbar = m_verticalScrollbar->isOverlayScrollbar(); |
| willRemoveScrollbar(m_verticalScrollbar.get(), VerticalScrollbar); |
| removeChild(m_verticalScrollbar.get()); |
| m_verticalScrollbar = 0; |
| if (contentSizeAffected) |
| *contentSizeAffected = !wasOverlayScrollbar; |
| return true; |
| } |
| |
| return false; |
| } |
| |
| PassRefPtr<Scrollbar> ScrollView::createScrollbar(ScrollbarOrientation orientation) |
| { |
| return Scrollbar::createNativeScrollbar(this, orientation, RegularScrollbar); |
| } |
| |
| void ScrollView::setScrollbarModes(ScrollbarMode horizontalMode, ScrollbarMode verticalMode, |
| bool horizontalLock, bool verticalLock) |
| { |
| bool needsUpdate = false; |
| |
| if (horizontalMode != horizontalScrollbarMode() && !m_horizontalScrollbarLock) { |
| m_horizontalScrollbarMode = horizontalMode; |
| needsUpdate = true; |
| } |
| |
| if (verticalMode != verticalScrollbarMode() && !m_verticalScrollbarLock) { |
| m_verticalScrollbarMode = verticalMode; |
| needsUpdate = true; |
| } |
| |
| if (horizontalLock) |
| setHorizontalScrollbarLock(); |
| |
| if (verticalLock) |
| setVerticalScrollbarLock(); |
| |
| if (!needsUpdate) |
| return; |
| |
| if (platformWidget()) |
| platformSetScrollbarModes(); |
| else |
| updateScrollbars(scrollOffset()); |
| } |
| |
| void ScrollView::scrollbarModes(ScrollbarMode& horizontalMode, ScrollbarMode& verticalMode) const |
| { |
| if (platformWidget()) { |
| platformScrollbarModes(horizontalMode, verticalMode); |
| return; |
| } |
| horizontalMode = m_horizontalScrollbarMode; |
| verticalMode = m_verticalScrollbarMode; |
| } |
| |
| void ScrollView::setCanHaveScrollbars(bool canScroll) |
| { |
| ScrollbarMode newHorizontalMode; |
| ScrollbarMode newVerticalMode; |
| |
| scrollbarModes(newHorizontalMode, newVerticalMode); |
| |
| if (canScroll && newVerticalMode == ScrollbarAlwaysOff) |
| newVerticalMode = ScrollbarAuto; |
| else if (!canScroll) |
| newVerticalMode = ScrollbarAlwaysOff; |
| |
| if (canScroll && newHorizontalMode == ScrollbarAlwaysOff) |
| newHorizontalMode = ScrollbarAuto; |
| else if (!canScroll) |
| newHorizontalMode = ScrollbarAlwaysOff; |
| |
| setScrollbarModes(newHorizontalMode, newVerticalMode); |
| } |
| |
| void ScrollView::setCanBlitOnScroll(bool b) |
| { |
| if (platformWidget()) { |
| platformSetCanBlitOnScroll(b); |
| return; |
| } |
| |
| m_canBlitOnScroll = b; |
| } |
| |
| bool ScrollView::canBlitOnScroll() const |
| { |
| if (platformWidget()) |
| return platformCanBlitOnScroll(); |
| |
| return m_canBlitOnScroll; |
| } |
| |
| void ScrollView::setPaintsEntireContents(bool paintsEntireContents) |
| { |
| m_paintsEntireContents = paintsEntireContents; |
| } |
| |
| void ScrollView::setClipsRepaints(bool clipsRepaints) |
| { |
| m_clipsRepaints = clipsRepaints; |
| } |
| |
| void ScrollView::setDelegatesScrolling(bool delegatesScrolling) |
| { |
| if (m_delegatesScrolling == delegatesScrolling) |
| return; |
| |
| m_delegatesScrolling = delegatesScrolling; |
| delegatesScrollingDidChange(); |
| } |
| |
| IntPoint ScrollView::contentsScrollPosition() const |
| { |
| #if PLATFORM(IOS) |
| if (platformWidget()) |
| return actualScrollPosition(); |
| #endif |
| return scrollPosition(); |
| } |
| |
| void ScrollView::setContentsScrollPosition(const IntPoint& position) |
| { |
| #if PLATFORM(IOS) |
| if (platformWidget()) |
| setActualScrollPosition(position); |
| #endif |
| setScrollPosition(position); |
| } |
| |
| #if !PLATFORM(IOS) |
| IntRect ScrollView::unobscuredContentRect(VisibleContentRectIncludesScrollbars scrollbarInclusion) const |
| { |
| return unobscuredContentRectInternal(scrollbarInclusion); |
| } |
| #endif |
| |
| IntRect ScrollView::unobscuredContentRectInternal(VisibleContentRectIncludesScrollbars scrollbarInclusion) const |
| { |
| FloatSize visibleContentSize = unscaledUnobscuredVisibleContentSize(scrollbarInclusion); |
| visibleContentSize.scale(1 / visibleContentScaleFactor()); |
| return IntRect(IntPoint(m_scrollOffset), expandedIntSize(visibleContentSize)); |
| } |
| |
| IntSize ScrollView::unscaledVisibleContentSizeIncludingObscuredArea(VisibleContentRectIncludesScrollbars scrollbarInclusion) const |
| { |
| if (platformWidget()) |
| return platformVisibleContentSizeIncludingObscuredArea(scrollbarInclusion == IncludeScrollbars); |
| |
| #if USE(TILED_BACKING_STORE) |
| if (!m_fixedVisibleContentRect.isEmpty()) |
| return m_fixedVisibleContentRect.size(); |
| #endif |
| |
| int verticalScrollbarWidth = 0; |
| int horizontalScrollbarHeight = 0; |
| |
| if (scrollbarInclusion == ExcludeScrollbars) { |
| if (Scrollbar* verticalBar = verticalScrollbar()) |
| verticalScrollbarWidth = !verticalBar->isOverlayScrollbar() ? verticalBar->width() : 0; |
| if (Scrollbar* horizontalBar = horizontalScrollbar()) |
| horizontalScrollbarHeight = !horizontalBar->isOverlayScrollbar() ? horizontalBar->height() : 0; |
| } |
| |
| return IntSize(width() - verticalScrollbarWidth, height() - horizontalScrollbarHeight).expandedTo(IntSize()); |
| } |
| |
| IntSize ScrollView::unscaledUnobscuredVisibleContentSize(VisibleContentRectIncludesScrollbars scrollbarInclusion) const |
| { |
| IntSize visibleContentSize = unscaledVisibleContentSizeIncludingObscuredArea(scrollbarInclusion); |
| |
| if (platformWidget()) |
| return platformVisibleContentSize(scrollbarInclusion == IncludeScrollbars); |
| |
| #if USE(TILED_BACKING_STORE) |
| if (!m_fixedVisibleContentRect.isEmpty()) |
| return visibleContentSize; |
| #endif |
| |
| visibleContentSize.setHeight(visibleContentSize.height() - topContentInset()); |
| return visibleContentSize; |
| } |
| |
| IntRect ScrollView::visibleContentRectInternal(VisibleContentRectIncludesScrollbars scrollbarInclusion, VisibleContentRectBehavior visibleContentRectBehavior) const |
| { |
| #if PLATFORM(IOS) |
| if (visibleContentRectBehavior == LegacyIOSDocumentViewRect) { |
| if (platformWidget()) |
| return platformVisibleContentRect(true /* include scrollbars */); |
| } |
| |
| if (platformWidget()) |
| return unobscuredContentRect(scrollbarInclusion); |
| #else |
| UNUSED_PARAM(visibleContentRectBehavior); |
| #endif |
| |
| if (platformWidget()) |
| return platformVisibleContentRect(scrollbarInclusion == IncludeScrollbars); |
| |
| #if USE(TILED_BACKING_STORE) |
| if (!m_fixedVisibleContentRect.isEmpty()) |
| return m_fixedVisibleContentRect; |
| #endif |
| |
| return unobscuredContentRect(scrollbarInclusion); |
| } |
| |
| IntSize ScrollView::layoutSize() const |
| { |
| return m_fixedLayoutSize.isEmpty() || !m_useFixedLayout ? unscaledUnobscuredVisibleContentSize(ExcludeScrollbars) : m_fixedLayoutSize; |
| } |
| |
| IntSize ScrollView::fixedLayoutSize() const |
| { |
| return m_fixedLayoutSize; |
| } |
| |
| void ScrollView::setFixedLayoutSize(const IntSize& newSize) |
| { |
| if (fixedLayoutSize() == newSize) |
| return; |
| m_fixedLayoutSize = newSize; |
| if (m_useFixedLayout) |
| fixedLayoutSizeChanged(); |
| } |
| |
| bool ScrollView::useFixedLayout() const |
| { |
| return m_useFixedLayout; |
| } |
| |
| void ScrollView::setUseFixedLayout(bool enable) |
| { |
| if (useFixedLayout() == enable) |
| return; |
| m_useFixedLayout = enable; |
| if (!m_fixedLayoutSize.isEmpty()) |
| fixedLayoutSizeChanged(); |
| } |
| |
| void ScrollView::fixedLayoutSizeChanged() |
| { |
| updateScrollbars(scrollOffset()); |
| contentsResized(); |
| } |
| |
| IntSize ScrollView::contentsSize() const |
| { |
| return m_contentsSize; |
| } |
| |
| void ScrollView::setContentsSize(const IntSize& newSize) |
| { |
| if (contentsSize() == newSize) |
| return; |
| m_contentsSize = newSize; |
| if (platformWidget()) |
| platformSetContentsSize(); |
| else |
| updateScrollbars(scrollOffset()); |
| updateOverhangAreas(); |
| } |
| |
| IntPoint ScrollView::maximumScrollPosition() const |
| { |
| IntPoint maximumOffset(contentsWidth() - visibleWidth() - scrollOrigin().x(), totalContentsSize().height() - visibleHeight() - scrollOrigin().y()); |
| maximumOffset.clampNegativeToZero(); |
| return maximumOffset; |
| } |
| |
| IntPoint ScrollView::minimumScrollPosition() const |
| { |
| return IntPoint(-scrollOrigin().x(), -scrollOrigin().y()); |
| } |
| |
| IntPoint ScrollView::adjustScrollPositionWithinRange(const IntPoint& scrollPoint) const |
| { |
| if (!constrainsScrollingToContentEdge()) |
| return scrollPoint; |
| |
| IntPoint newScrollPosition = scrollPoint.shrunkTo(maximumScrollPosition()); |
| newScrollPosition = newScrollPosition.expandedTo(minimumScrollPosition()); |
| return newScrollPosition; |
| } |
| |
| IntSize ScrollView::documentScrollOffsetRelativeToViewOrigin() const |
| { |
| return scrollOffset() - IntSize(0, headerHeight() + topContentInset(TopContentInsetType::WebCoreOrPlatformContentInset)); |
| } |
| |
| IntPoint ScrollView::documentScrollPositionRelativeToViewOrigin() const |
| { |
| IntPoint scrollPosition = this->scrollPosition(); |
| return IntPoint(scrollPosition.x(), scrollPosition.y() - headerHeight() - topContentInset(TopContentInsetType::WebCoreOrPlatformContentInset)); |
| } |
| |
| IntSize ScrollView::documentScrollOffsetRelativeToScrollableAreaOrigin() const |
| { |
| return scrollOffset() - IntSize(0, headerHeight()); |
| } |
| |
| int ScrollView::scrollSize(ScrollbarOrientation orientation) const |
| { |
| // If no scrollbars are present, it does not indicate content is not be scrollable. |
| if (!m_horizontalScrollbar && !m_verticalScrollbar && !prohibitsScrolling()) { |
| IntSize scrollSize = m_contentsSize - visibleContentRect(LegacyIOSDocumentVisibleRect).size(); |
| scrollSize.clampNegativeToZero(); |
| return orientation == HorizontalScrollbar ? scrollSize.width() : scrollSize.height(); |
| } |
| |
| Scrollbar* scrollbar = ((orientation == HorizontalScrollbar) ? m_horizontalScrollbar : m_verticalScrollbar).get(); |
| return scrollbar ? (scrollbar->totalSize() - scrollbar->visibleSize()) : 0; |
| } |
| |
| void ScrollView::notifyPageThatContentAreaWillPaint() const |
| { |
| } |
| |
| void ScrollView::setScrollOffset(const IntPoint& offset) |
| { |
| int horizontalOffset = offset.x(); |
| int verticalOffset = offset.y(); |
| if (constrainsScrollingToContentEdge()) { |
| horizontalOffset = std::max(std::min(horizontalOffset, contentsWidth() - visibleWidth()), 0); |
| verticalOffset = std::max(std::min(verticalOffset, totalContentsSize().height() - visibleHeight()), 0); |
| } |
| |
| IntSize newOffset = m_scrollOffset; |
| newOffset.setWidth(horizontalOffset - scrollOrigin().x()); |
| newOffset.setHeight(verticalOffset - scrollOrigin().y()); |
| |
| scrollTo(newOffset); |
| } |
| |
| void ScrollView::scrollTo(const IntSize& newOffset) |
| { |
| IntSize scrollDelta = newOffset - m_scrollOffset; |
| if (scrollDelta == IntSize()) |
| return; |
| m_scrollOffset = newOffset; |
| |
| if (scrollbarsSuppressed()) |
| return; |
| |
| #if USE(TILED_BACKING_STORE) |
| if (delegatesScrolling()) { |
| hostWindow()->delegatedScrollRequested(IntPoint(newOffset)); |
| return; |
| } |
| #endif |
| updateLayerPositionsAfterScrolling(); |
| scrollContents(scrollDelta); |
| updateCompositingLayersAfterScrolling(); |
| } |
| |
| int ScrollView::scrollPosition(Scrollbar* scrollbar) const |
| { |
| if (scrollbar->orientation() == HorizontalScrollbar) |
| return scrollPosition().x() + scrollOrigin().x(); |
| if (scrollbar->orientation() == VerticalScrollbar) |
| return scrollPosition().y() + scrollOrigin().y(); |
| return 0; |
| } |
| |
| void ScrollView::setScrollPosition(const IntPoint& scrollPoint) |
| { |
| if (prohibitsScrolling()) |
| return; |
| |
| if (platformWidget()) { |
| platformSetScrollPosition(scrollPoint); |
| return; |
| } |
| |
| IntPoint newScrollPosition = !delegatesScrolling() ? adjustScrollPositionWithinRange(scrollPoint) : scrollPoint; |
| |
| if ((!delegatesScrolling() || !inProgrammaticScroll()) && newScrollPosition == scrollPosition()) |
| return; |
| |
| if (requestScrollPositionUpdate(newScrollPosition)) |
| return; |
| |
| updateScrollbars(IntSize(newScrollPosition.x(), newScrollPosition.y())); |
| } |
| |
| bool ScrollView::scroll(ScrollDirection direction, ScrollGranularity granularity) |
| { |
| if (platformWidget()) |
| return platformScroll(direction, granularity); |
| |
| return ScrollableArea::scroll(direction, granularity); |
| } |
| |
| bool ScrollView::logicalScroll(ScrollLogicalDirection direction, ScrollGranularity granularity) |
| { |
| return scroll(logicalToPhysical(direction, isVerticalDocument(), isFlippedDocument()), granularity); |
| } |
| |
| IntSize ScrollView::overhangAmount() const |
| { |
| IntSize stretch; |
| |
| int physicalScrollY = scrollPosition().y() + scrollOrigin().y(); |
| if (physicalScrollY < 0) |
| stretch.setHeight(physicalScrollY); |
| else if (totalContentsSize().height() && physicalScrollY > totalContentsSize().height() - visibleHeight()) |
| stretch.setHeight(physicalScrollY - (totalContentsSize().height() - visibleHeight())); |
| |
| int physicalScrollX = scrollPosition().x() + scrollOrigin().x(); |
| if (physicalScrollX < 0) |
| stretch.setWidth(physicalScrollX); |
| else if (contentsWidth() && physicalScrollX > contentsWidth() - visibleWidth()) |
| stretch.setWidth(physicalScrollX - (contentsWidth() - visibleWidth())); |
| |
| return stretch; |
| } |
| |
| void ScrollView::windowResizerRectChanged() |
| { |
| if (platformWidget()) |
| return; |
| |
| updateScrollbars(scrollOffset()); |
| } |
| |
| void ScrollView::updateScrollbars(const IntSize& desiredOffset) |
| { |
| if (m_inUpdateScrollbars || prohibitsScrolling() || platformWidget() || delegatesScrolling()) |
| return; |
| |
| bool hasOverlayScrollbars = (!m_horizontalScrollbar || m_horizontalScrollbar->isOverlayScrollbar()) && (!m_verticalScrollbar || m_verticalScrollbar->isOverlayScrollbar()); |
| |
| // If we came in here with the view already needing a layout, then go ahead and do that |
| // first. (This will be the common case, e.g., when the page changes due to window resizing for example). |
| // This layout will not re-enter updateScrollbars and does not count towards our max layout pass total. |
| if (!m_scrollbarsSuppressed && !hasOverlayScrollbars) { |
| m_inUpdateScrollbars = true; |
| visibleContentsResized(); |
| m_inUpdateScrollbars = false; |
| } |
| |
| IntRect oldScrollCornerRect = scrollCornerRect(); |
| |
| bool hasHorizontalScrollbar = m_horizontalScrollbar; |
| bool hasVerticalScrollbar = m_verticalScrollbar; |
| |
| bool newHasHorizontalScrollbar = hasHorizontalScrollbar; |
| bool newHasVerticalScrollbar = hasVerticalScrollbar; |
| |
| ScrollbarMode hScroll = m_horizontalScrollbarMode; |
| ScrollbarMode vScroll = m_verticalScrollbarMode; |
| |
| if (hScroll != ScrollbarAuto) |
| newHasHorizontalScrollbar = (hScroll == ScrollbarAlwaysOn); |
| if (vScroll != ScrollbarAuto) |
| newHasVerticalScrollbar = (vScroll == ScrollbarAlwaysOn); |
| |
| bool scrollbarAddedOrRemoved = false; |
| |
| if (m_scrollbarsSuppressed || (hScroll != ScrollbarAuto && vScroll != ScrollbarAuto)) { |
| if (hasHorizontalScrollbar != newHasHorizontalScrollbar && (hasHorizontalScrollbar || !avoidScrollbarCreation())) { |
| if (setHasHorizontalScrollbar(newHasHorizontalScrollbar)) |
| scrollbarAddedOrRemoved = true; |
| } |
| |
| if (hasVerticalScrollbar != newHasVerticalScrollbar && (hasVerticalScrollbar || !avoidScrollbarCreation())) { |
| if (setHasVerticalScrollbar(newHasVerticalScrollbar)) |
| scrollbarAddedOrRemoved = true; |
| } |
| } else { |
| bool sendContentResizedNotification = false; |
| |
| IntSize docSize = totalContentsSize(); |
| IntSize fullVisibleSize = unobscuredContentRectIncludingScrollbars().size(); |
| |
| if (hScroll == ScrollbarAuto) |
| newHasHorizontalScrollbar = docSize.width() > visibleWidth(); |
| if (vScroll == ScrollbarAuto) |
| newHasVerticalScrollbar = docSize.height() > visibleHeight(); |
| |
| bool needAnotherPass = false; |
| if (!hasOverlayScrollbars) { |
| // If we ever turn one scrollbar off, always turn the other one off too. Never ever |
| // try to both gain/lose a scrollbar in the same pass. |
| if (!m_updateScrollbarsPass && docSize.width() <= fullVisibleSize.width() && docSize.height() <= fullVisibleSize.height()) { |
| if (hScroll == ScrollbarAuto) |
| newHasHorizontalScrollbar = false; |
| if (vScroll == ScrollbarAuto) |
| newHasVerticalScrollbar = false; |
| } |
| if (!newHasHorizontalScrollbar && hasHorizontalScrollbar && vScroll != ScrollbarAlwaysOn) { |
| newHasVerticalScrollbar = false; |
| needAnotherPass = true; |
| } |
| if (!newHasVerticalScrollbar && hasVerticalScrollbar && hScroll != ScrollbarAlwaysOn) { |
| newHasHorizontalScrollbar = false; |
| needAnotherPass = true; |
| } |
| } |
| |
| if (hasHorizontalScrollbar != newHasHorizontalScrollbar && (hasHorizontalScrollbar || !avoidScrollbarCreation())) { |
| if (scrollOrigin().y() && !newHasHorizontalScrollbar) |
| ScrollableArea::setScrollOrigin(IntPoint(scrollOrigin().x(), scrollOrigin().y() - m_horizontalScrollbar->height())); |
| if (m_horizontalScrollbar) |
| m_horizontalScrollbar->invalidate(); |
| |
| bool changeAffectsContentSize = false; |
| if (setHasHorizontalScrollbar(newHasHorizontalScrollbar, &changeAffectsContentSize)) { |
| scrollbarAddedOrRemoved = true; |
| sendContentResizedNotification |= changeAffectsContentSize; |
| } |
| } |
| |
| if (hasVerticalScrollbar != newHasVerticalScrollbar && (hasVerticalScrollbar || !avoidScrollbarCreation())) { |
| if (scrollOrigin().x() && !newHasVerticalScrollbar) |
| ScrollableArea::setScrollOrigin(IntPoint(scrollOrigin().x() - m_verticalScrollbar->width(), scrollOrigin().y())); |
| if (m_verticalScrollbar) |
| m_verticalScrollbar->invalidate(); |
| |
| bool changeAffectsContentSize = false; |
| if (setHasVerticalScrollbar(newHasVerticalScrollbar, &changeAffectsContentSize)) { |
| scrollbarAddedOrRemoved = true; |
| sendContentResizedNotification |= changeAffectsContentSize; |
| } |
| } |
| |
| const unsigned cMaxUpdateScrollbarsPass = 2; |
| if ((sendContentResizedNotification || needAnotherPass) && m_updateScrollbarsPass < cMaxUpdateScrollbarsPass) { |
| m_updateScrollbarsPass++; |
| contentsResized(); |
| visibleContentsResized(); |
| IntSize newDocSize = totalContentsSize(); |
| if (newDocSize == docSize) { |
| // The layout with the new scroll state had no impact on |
| // the document's overall size, so updateScrollbars didn't get called. |
| // Recur manually. |
| updateScrollbars(desiredOffset); |
| } |
| m_updateScrollbarsPass--; |
| } |
| } |
| |
| if (scrollbarAddedOrRemoved) |
| addedOrRemovedScrollbar(); |
| |
| // Set up the range (and page step/line step), but only do this if we're not in a nested call (to avoid |
| // doing it multiple times). |
| if (m_updateScrollbarsPass) |
| return; |
| |
| m_inUpdateScrollbars = true; |
| |
| if (m_horizontalScrollbar) { |
| int clientWidth = visibleWidth(); |
| int pageStep = Scrollbar::pageStep(clientWidth); |
| IntRect oldRect(m_horizontalScrollbar->frameRect()); |
| IntRect hBarRect(0, |
| height() - m_horizontalScrollbar->height(), |
| width() - (m_verticalScrollbar ? m_verticalScrollbar->width() : 0), |
| m_horizontalScrollbar->height()); |
| m_horizontalScrollbar->setFrameRect(hBarRect); |
| if (!m_scrollbarsSuppressed && oldRect != m_horizontalScrollbar->frameRect()) |
| m_horizontalScrollbar->invalidate(); |
| |
| if (m_scrollbarsSuppressed) |
| m_horizontalScrollbar->setSuppressInvalidation(true); |
| m_horizontalScrollbar->setEnabled(contentsWidth() > clientWidth); |
| m_horizontalScrollbar->setSteps(Scrollbar::pixelsPerLineStep(), pageStep); |
| m_horizontalScrollbar->setProportion(clientWidth, contentsWidth()); |
| if (m_scrollbarsSuppressed) |
| m_horizontalScrollbar->setSuppressInvalidation(false); |
| } |
| |
| if (m_verticalScrollbar) { |
| int clientHeight = visibleHeight(); |
| int pageStep = Scrollbar::pageStep(clientHeight); |
| IntRect oldRect(m_verticalScrollbar->frameRect()); |
| IntRect vBarRect(width() - m_verticalScrollbar->width(), |
| topContentInset(), |
| m_verticalScrollbar->width(), |
| height() - topContentInset() - (m_horizontalScrollbar ? m_horizontalScrollbar->height() : 0)); |
| m_verticalScrollbar->setFrameRect(vBarRect); |
| if (!m_scrollbarsSuppressed && oldRect != m_verticalScrollbar->frameRect()) |
| m_verticalScrollbar->invalidate(); |
| |
| if (m_scrollbarsSuppressed) |
| m_verticalScrollbar->setSuppressInvalidation(true); |
| m_verticalScrollbar->setEnabled(totalContentsSize().height() > clientHeight); |
| m_verticalScrollbar->setSteps(Scrollbar::pixelsPerLineStep(), pageStep); |
| m_verticalScrollbar->setProportion(clientHeight, totalContentsSize().height()); |
| if (m_scrollbarsSuppressed) |
| m_verticalScrollbar->setSuppressInvalidation(false); |
| } |
| |
| if (hasHorizontalScrollbar != newHasHorizontalScrollbar || hasVerticalScrollbar != newHasVerticalScrollbar) { |
| // FIXME: Is frameRectsChanged really necessary here? Have any frame rects changed? |
| frameRectsChanged(); |
| positionScrollbarLayers(); |
| updateScrollCorner(); |
| if (!m_horizontalScrollbar && !m_verticalScrollbar) |
| invalidateScrollCornerRect(oldScrollCornerRect); |
| } |
| |
| IntPoint adjustedScrollPosition = IntPoint(desiredOffset); |
| if (!isRubberBandInProgress()) |
| adjustedScrollPosition = adjustScrollPositionWithinRange(adjustedScrollPosition); |
| |
| if (adjustedScrollPosition != scrollPosition() || scrollOriginChanged()) { |
| ScrollableArea::scrollToOffsetWithoutAnimation(adjustedScrollPosition + toIntSize(scrollOrigin())); |
| resetScrollOriginChanged(); |
| } |
| |
| // Make sure the scrollbar offsets are up to date. |
| if (m_horizontalScrollbar) |
| m_horizontalScrollbar->offsetDidChange(); |
| if (m_verticalScrollbar) |
| m_verticalScrollbar->offsetDidChange(); |
| |
| m_inUpdateScrollbars = false; |
| } |
| |
| const int panIconSizeLength = 16; |
| |
| IntRect ScrollView::rectToCopyOnScroll() const |
| { |
| IntRect scrollViewRect = convertToRootView(IntRect(0, 0, visibleWidth(), visibleHeight())); |
| if (hasOverlayScrollbars()) { |
| int verticalScrollbarWidth = (verticalScrollbar() && !hasLayerForVerticalScrollbar()) ? verticalScrollbar()->width() : 0; |
| int horizontalScrollbarHeight = (horizontalScrollbar() && !hasLayerForHorizontalScrollbar()) ? horizontalScrollbar()->height() : 0; |
| |
| scrollViewRect.setWidth(scrollViewRect.width() - verticalScrollbarWidth); |
| scrollViewRect.setHeight(scrollViewRect.height() - horizontalScrollbarHeight); |
| } |
| return scrollViewRect; |
| } |
| |
| void ScrollView::scrollContents(const IntSize& scrollDelta) |
| { |
| HostWindow* window = hostWindow(); |
| if (!window) |
| return; |
| |
| // Since scrolling is double buffered, we will be blitting the scroll view's intersection |
| // with the clip rect every time to keep it smooth. |
| IntRect clipRect = windowClipRect(); |
| IntRect scrollViewRect = rectToCopyOnScroll(); |
| IntRect updateRect = clipRect; |
| updateRect.intersect(scrollViewRect); |
| |
| // Invalidate the root view (not the backing store). |
| window->invalidateRootView(updateRect); |
| |
| if (m_drawPanScrollIcon) { |
| // FIXME: the pan icon is broken when accelerated compositing is on, since it will draw under the compositing layers. |
| // https://bugs.webkit.org/show_bug.cgi?id=47837 |
| int panIconDirtySquareSizeLength = 2 * (panIconSizeLength + std::max(abs(scrollDelta.width()), abs(scrollDelta.height()))); // We only want to repaint what's necessary |
| IntPoint panIconDirtySquareLocation = IntPoint(m_panScrollIconPoint.x() - (panIconDirtySquareSizeLength / 2), m_panScrollIconPoint.y() - (panIconDirtySquareSizeLength / 2)); |
| IntRect panScrollIconDirtyRect = IntRect(panIconDirtySquareLocation, IntSize(panIconDirtySquareSizeLength, panIconDirtySquareSizeLength)); |
| panScrollIconDirtyRect.intersect(clipRect); |
| window->invalidateContentsAndRootView(panScrollIconDirtyRect); |
| } |
| |
| if (canBlitOnScroll()) { // The main frame can just blit the WebView window |
| // FIXME: Find a way to scroll subframes with this faster path |
| if (!scrollContentsFastPath(-scrollDelta, scrollViewRect, clipRect)) |
| scrollContentsSlowPath(updateRect); |
| } else { |
| // We need to go ahead and repaint the entire backing store. Do it now before moving the |
| // windowed plugins. |
| scrollContentsSlowPath(updateRect); |
| } |
| |
| // Invalidate the overhang areas if they are visible. |
| updateOverhangAreas(); |
| |
| // This call will move children with native widgets (plugins) and invalidate them as well. |
| frameRectsChanged(); |
| |
| // Now blit the backingstore into the window which should be very fast. |
| window->invalidateRootView(IntRect()); |
| } |
| |
| bool ScrollView::scrollContentsFastPath(const IntSize& scrollDelta, const IntRect& rectToScroll, const IntRect& clipRect) |
| { |
| hostWindow()->scroll(scrollDelta, rectToScroll, clipRect); |
| return true; |
| } |
| |
| void ScrollView::scrollContentsSlowPath(const IntRect& updateRect) |
| { |
| hostWindow()->invalidateContentsForSlowScroll(updateRect); |
| } |
| |
| IntPoint ScrollView::rootViewToContents(const IntPoint& rootViewPoint) const |
| { |
| if (delegatesScrolling()) |
| return convertFromRootView(rootViewPoint); |
| |
| IntPoint viewPoint = convertFromRootView(rootViewPoint); |
| return viewPoint + documentScrollOffsetRelativeToViewOrigin(); |
| } |
| |
| IntPoint ScrollView::contentsToRootView(const IntPoint& contentsPoint) const |
| { |
| if (delegatesScrolling()) |
| return convertToRootView(contentsPoint); |
| |
| IntPoint viewPoint = contentsPoint + IntSize(0, headerHeight() + topContentInset(TopContentInsetType::WebCoreOrPlatformContentInset)) - scrollOffset(); |
| return convertToRootView(viewPoint); |
| } |
| |
| IntRect ScrollView::rootViewToContents(const IntRect& rootViewRect) const |
| { |
| if (delegatesScrolling()) |
| return convertFromRootView(rootViewRect); |
| |
| IntRect viewRect = convertFromRootView(rootViewRect); |
| viewRect.move(documentScrollOffsetRelativeToViewOrigin()); |
| return viewRect; |
| } |
| |
| IntRect ScrollView::contentsToRootView(const IntRect& contentsRect) const |
| { |
| if (delegatesScrolling()) |
| return convertToRootView(contentsRect); |
| |
| IntRect viewRect = contentsRect; |
| viewRect.move(-scrollOffset() + IntSize(0, headerHeight() + topContentInset(TopContentInsetType::WebCoreOrPlatformContentInset))); |
| return convertToRootView(viewRect); |
| } |
| |
| IntPoint ScrollView::rootViewToTotalContents(const IntPoint& rootViewPoint) const |
| { |
| if (delegatesScrolling()) |
| return convertFromRootView(rootViewPoint); |
| |
| IntPoint viewPoint = convertFromRootView(rootViewPoint); |
| return viewPoint + scrollOffset() - IntSize(0, topContentInset(TopContentInsetType::WebCoreOrPlatformContentInset)); |
| } |
| |
| IntPoint ScrollView::windowToContents(const IntPoint& windowPoint) const |
| { |
| if (delegatesScrolling()) |
| return convertFromContainingWindow(windowPoint); |
| |
| IntPoint viewPoint = convertFromContainingWindow(windowPoint); |
| return viewPoint + documentScrollOffsetRelativeToViewOrigin(); |
| } |
| |
| IntPoint ScrollView::contentsToWindow(const IntPoint& contentsPoint) const |
| { |
| if (delegatesScrolling()) |
| return convertToContainingWindow(contentsPoint); |
| |
| IntPoint viewPoint = contentsPoint + IntSize(0, headerHeight() + topContentInset(TopContentInsetType::WebCoreOrPlatformContentInset)) - scrollOffset(); |
| return convertToContainingWindow(viewPoint); |
| } |
| |
| IntRect ScrollView::windowToContents(const IntRect& windowRect) const |
| { |
| if (delegatesScrolling()) |
| return convertFromContainingWindow(windowRect); |
| |
| IntRect viewRect = convertFromContainingWindow(windowRect); |
| viewRect.move(documentScrollOffsetRelativeToViewOrigin()); |
| return viewRect; |
| } |
| |
| IntRect ScrollView::contentsToWindow(const IntRect& contentsRect) const |
| { |
| if (delegatesScrolling()) |
| return convertToContainingWindow(contentsRect); |
| |
| IntRect viewRect = contentsRect; |
| viewRect.move(-scrollOffset() + IntSize(0, headerHeight() + topContentInset(TopContentInsetType::WebCoreOrPlatformContentInset))); |
| return convertToContainingWindow(viewRect); |
| } |
| |
| IntRect ScrollView::contentsToScreen(const IntRect& rect) const |
| { |
| HostWindow* window = hostWindow(); |
| if (platformWidget()) |
| return platformContentsToScreen(rect); |
| if (!window) |
| return IntRect(); |
| return window->rootViewToScreen(contentsToRootView(rect)); |
| } |
| |
| IntPoint ScrollView::screenToContents(const IntPoint& point) const |
| { |
| HostWindow* window = hostWindow(); |
| if (platformWidget()) |
| return platformScreenToContents(point); |
| if (!window) |
| return IntPoint(); |
| return rootViewToContents(window->screenToRootView(point)); |
| } |
| |
| bool ScrollView::containsScrollbarsAvoidingResizer() const |
| { |
| return !m_scrollbarsAvoidingResizer; |
| } |
| |
| void ScrollView::adjustScrollbarsAvoidingResizerCount(int overlapDelta) |
| { |
| int oldCount = m_scrollbarsAvoidingResizer; |
| m_scrollbarsAvoidingResizer += overlapDelta; |
| if (parent()) |
| parent()->adjustScrollbarsAvoidingResizerCount(overlapDelta); |
| else if (!scrollbarsSuppressed()) { |
| // If we went from n to 0 or from 0 to n and we're the outermost view, |
| // we need to invalidate the windowResizerRect(), since it will now need to paint |
| // differently. |
| if ((oldCount > 0 && m_scrollbarsAvoidingResizer == 0) || |
| (oldCount == 0 && m_scrollbarsAvoidingResizer > 0)) |
| invalidateRect(windowResizerRect()); |
| } |
| } |
| |
| void ScrollView::setParent(ScrollView* parentView) |
| { |
| if (parentView == parent()) |
| return; |
| |
| if (m_scrollbarsAvoidingResizer && parent()) |
| parent()->adjustScrollbarsAvoidingResizerCount(-m_scrollbarsAvoidingResizer); |
| |
| Widget::setParent(parentView); |
| |
| if (m_scrollbarsAvoidingResizer && parent()) |
| parent()->adjustScrollbarsAvoidingResizerCount(m_scrollbarsAvoidingResizer); |
| } |
| |
| void ScrollView::setScrollbarsSuppressed(bool suppressed, bool repaintOnUnsuppress) |
| { |
| if (suppressed == m_scrollbarsSuppressed) |
| return; |
| |
| m_scrollbarsSuppressed = suppressed; |
| |
| if (platformWidget()) |
| platformSetScrollbarsSuppressed(repaintOnUnsuppress); |
| else if (repaintOnUnsuppress && !suppressed) { |
| if (m_horizontalScrollbar) |
| m_horizontalScrollbar->invalidate(); |
| if (m_verticalScrollbar) |
| m_verticalScrollbar->invalidate(); |
| |
| // Invalidate the scroll corner too on unsuppress. |
| invalidateRect(scrollCornerRect()); |
| } |
| } |
| |
| Scrollbar* ScrollView::scrollbarAtPoint(const IntPoint& windowPoint) |
| { |
| if (platformWidget()) |
| return 0; |
| |
| IntPoint viewPoint = convertFromContainingWindow(windowPoint); |
| if (m_horizontalScrollbar && m_horizontalScrollbar->shouldParticipateInHitTesting() && m_horizontalScrollbar->frameRect().contains(viewPoint)) |
| return m_horizontalScrollbar.get(); |
| if (m_verticalScrollbar && m_verticalScrollbar->shouldParticipateInHitTesting() && m_verticalScrollbar->frameRect().contains(viewPoint)) |
| return m_verticalScrollbar.get(); |
| return 0; |
| } |
| |
| void ScrollView::setScrollbarOverlayStyle(ScrollbarOverlayStyle overlayStyle) |
| { |
| ScrollableArea::setScrollbarOverlayStyle(overlayStyle); |
| platformSetScrollbarOverlayStyle(overlayStyle); |
| } |
| |
| void ScrollView::setFrameRect(const IntRect& newRect) |
| { |
| IntRect oldRect = frameRect(); |
| |
| if (newRect == oldRect) |
| return; |
| |
| Widget::setFrameRect(newRect); |
| |
| frameRectsChanged(); |
| |
| updateScrollbars(scrollOffset()); |
| |
| if (!m_useFixedLayout && oldRect.size() != newRect.size()) |
| contentsResized(); |
| } |
| |
| void ScrollView::frameRectsChanged() |
| { |
| if (platformWidget()) |
| return; |
| |
| HashSet<RefPtr<Widget>>::const_iterator end = m_children.end(); |
| for (HashSet<RefPtr<Widget>>::const_iterator current = m_children.begin(); current != end; ++current) |
| (*current)->frameRectsChanged(); |
| } |
| |
| void ScrollView::clipRectChanged() |
| { |
| HashSet<RefPtr<Widget>>::const_iterator end = m_children.end(); |
| for (HashSet<RefPtr<Widget>>::const_iterator current = m_children.begin(); current != end; ++current) |
| (*current)->clipRectChanged(); |
| } |
| |
| static void positionScrollbarLayer(GraphicsLayer* graphicsLayer, Scrollbar* scrollbar) |
| { |
| if (!graphicsLayer || !scrollbar) |
| return; |
| |
| IntRect scrollbarRect = scrollbar->frameRect(); |
| graphicsLayer->setPosition(scrollbarRect.location()); |
| |
| if (scrollbarRect.size() == graphicsLayer->size()) |
| return; |
| |
| graphicsLayer->setSize(scrollbarRect.size()); |
| |
| if (graphicsLayer->usesContentsLayer()) { |
| graphicsLayer->setContentsRect(IntRect(0, 0, scrollbarRect.width(), scrollbarRect.height())); |
| return; |
| } |
| |
| graphicsLayer->setDrawsContent(true); |
| graphicsLayer->setNeedsDisplay(); |
| } |
| |
| static void positionScrollCornerLayer(GraphicsLayer* graphicsLayer, const IntRect& cornerRect) |
| { |
| if (!graphicsLayer) |
| return; |
| graphicsLayer->setDrawsContent(!cornerRect.isEmpty()); |
| graphicsLayer->setPosition(cornerRect.location()); |
| if (cornerRect.size() != graphicsLayer->size()) |
| graphicsLayer->setNeedsDisplay(); |
| graphicsLayer->setSize(cornerRect.size()); |
| } |
| |
| void ScrollView::positionScrollbarLayers() |
| { |
| positionScrollbarLayer(layerForHorizontalScrollbar(), horizontalScrollbar()); |
| positionScrollbarLayer(layerForVerticalScrollbar(), verticalScrollbar()); |
| positionScrollCornerLayer(layerForScrollCorner(), scrollCornerRect()); |
| } |
| |
| void ScrollView::repaintContentRectangle(const IntRect& rect) |
| { |
| IntRect paintRect = rect; |
| if (clipsRepaints() && !paintsEntireContents()) |
| paintRect.intersect(visibleContentRect(LegacyIOSDocumentVisibleRect)); |
| if (paintRect.isEmpty()) |
| return; |
| |
| if (platformWidget()) { |
| notifyPageThatContentAreaWillPaint(); |
| platformRepaintContentRectangle(paintRect); |
| return; |
| } |
| |
| if (HostWindow* window = hostWindow()) |
| window->invalidateContentsAndRootView(contentsToWindow(paintRect)); |
| } |
| |
| IntRect ScrollView::scrollCornerRect() const |
| { |
| IntRect cornerRect; |
| |
| if (hasOverlayScrollbars()) |
| return cornerRect; |
| |
| int heightTrackedByScrollbar = height() - topContentInset(); |
| |
| if (m_horizontalScrollbar && width() - m_horizontalScrollbar->width() > 0) { |
| cornerRect.unite(IntRect(m_horizontalScrollbar->width(), |
| height() - m_horizontalScrollbar->height(), |
| width() - m_horizontalScrollbar->width(), |
| m_horizontalScrollbar->height())); |
| } |
| |
| if (m_verticalScrollbar && heightTrackedByScrollbar - m_verticalScrollbar->height() > 0) { |
| cornerRect.unite(IntRect(width() - m_verticalScrollbar->width(), |
| m_verticalScrollbar->height() + topContentInset(), |
| m_verticalScrollbar->width(), |
| heightTrackedByScrollbar - m_verticalScrollbar->height())); |
| } |
| |
| return cornerRect; |
| } |
| |
| bool ScrollView::isScrollCornerVisible() const |
| { |
| return !scrollCornerRect().isEmpty(); |
| } |
| |
| void ScrollView::scrollbarStyleChanged(int, bool forceUpdate) |
| { |
| if (!forceUpdate) |
| return; |
| |
| contentsResized(); |
| updateScrollbars(scrollOffset()); |
| positionScrollbarLayers(); |
| } |
| |
| void ScrollView::updateScrollCorner() |
| { |
| } |
| |
| void ScrollView::paintScrollCorner(GraphicsContext* context, const IntRect& cornerRect) |
| { |
| ScrollbarTheme::theme()->paintScrollCorner(this, context, cornerRect); |
| } |
| |
| void ScrollView::paintScrollbar(GraphicsContext* context, Scrollbar* bar, const IntRect& rect) |
| { |
| bar->paint(context, rect); |
| } |
| |
| void ScrollView::invalidateScrollCornerRect(const IntRect& rect) |
| { |
| invalidateRect(rect); |
| } |
| |
| void ScrollView::paintScrollbars(GraphicsContext* context, const IntRect& rect) |
| { |
| if (m_horizontalScrollbar && !layerForHorizontalScrollbar()) |
| paintScrollbar(context, m_horizontalScrollbar.get(), rect); |
| if (m_verticalScrollbar && !layerForVerticalScrollbar()) |
| paintScrollbar(context, m_verticalScrollbar.get(), rect); |
| |
| if (layerForScrollCorner()) |
| return; |
| |
| paintScrollCorner(context, scrollCornerRect()); |
| } |
| |
| void ScrollView::paintPanScrollIcon(GraphicsContext* context) |
| { |
| static Image* panScrollIcon = Image::loadPlatformResource("panIcon").leakRef(); |
| IntPoint iconGCPoint = m_panScrollIconPoint; |
| if (parent()) |
| iconGCPoint = parent()->windowToContents(iconGCPoint); |
| context->drawImage(panScrollIcon, ColorSpaceDeviceRGB, iconGCPoint); |
| } |
| |
| void ScrollView::paint(GraphicsContext* context, const IntRect& rect) |
| { |
| if (platformWidget()) { |
| Widget::paint(context, rect); |
| return; |
| } |
| |
| if (context->paintingDisabled() && !context->updatingControlTints()) |
| return; |
| |
| notifyPageThatContentAreaWillPaint(); |
| |
| IntRect documentDirtyRect = rect; |
| if (!paintsEntireContents()) { |
| IntRect visibleAreaWithoutScrollbars(location(), visibleContentRect(LegacyIOSDocumentVisibleRect).size()); |
| documentDirtyRect.intersect(visibleAreaWithoutScrollbars); |
| } |
| |
| if (!documentDirtyRect.isEmpty()) { |
| GraphicsContextStateSaver stateSaver(*context); |
| |
| context->translate(x(), y()); |
| documentDirtyRect.moveBy(-location()); |
| |
| if (!paintsEntireContents()) { |
| context->translate(-scrollX(), -scrollY()); |
| documentDirtyRect.moveBy(scrollPosition()); |
| |
| context->clip(visibleContentRect(LegacyIOSDocumentVisibleRect)); |
| } |
| |
| paintContents(context, documentDirtyRect); |
| } |
| |
| #if ENABLE(RUBBER_BANDING) |
| if (!layerForOverhangAreas()) |
| calculateAndPaintOverhangAreas(context, rect); |
| #else |
| calculateAndPaintOverhangAreas(context, rect); |
| #endif |
| |
| // Now paint the scrollbars. |
| if (!m_scrollbarsSuppressed && (m_horizontalScrollbar || m_verticalScrollbar)) { |
| GraphicsContextStateSaver stateSaver(*context); |
| IntRect scrollViewDirtyRect = rect; |
| IntRect visibleAreaWithScrollbars(location(), unobscuredContentRectIncludingScrollbars().size()); |
| scrollViewDirtyRect.intersect(visibleAreaWithScrollbars); |
| context->translate(x(), y()); |
| scrollViewDirtyRect.moveBy(-location()); |
| context->clip(IntRect(IntPoint(), visibleAreaWithScrollbars.size())); |
| |
| paintScrollbars(context, scrollViewDirtyRect); |
| } |
| |
| // Paint the panScroll Icon |
| if (m_drawPanScrollIcon) |
| paintPanScrollIcon(context); |
| } |
| |
| void ScrollView::calculateOverhangAreasForPainting(IntRect& horizontalOverhangRect, IntRect& verticalOverhangRect) |
| { |
| int verticalScrollbarWidth = (verticalScrollbar() && !verticalScrollbar()->isOverlayScrollbar()) |
| ? verticalScrollbar()->width() : 0; |
| int horizontalScrollbarHeight = (horizontalScrollbar() && !horizontalScrollbar()->isOverlayScrollbar()) |
| ? horizontalScrollbar()->height() : 0; |
| |
| int physicalScrollY = scrollPosition().y() + scrollOrigin().y(); |
| if (physicalScrollY < 0) { |
| horizontalOverhangRect = frameRect(); |
| horizontalOverhangRect.setHeight(-physicalScrollY); |
| horizontalOverhangRect.setWidth(horizontalOverhangRect.width() - verticalScrollbarWidth); |
| } else if (totalContentsSize().height() && physicalScrollY > totalContentsSize().height() - visibleHeight()) { |
| int height = physicalScrollY - (totalContentsSize().height() - visibleHeight()); |
| horizontalOverhangRect = frameRect(); |
| horizontalOverhangRect.setY(frameRect().maxY() - height - horizontalScrollbarHeight); |
| horizontalOverhangRect.setHeight(height); |
| horizontalOverhangRect.setWidth(horizontalOverhangRect.width() - verticalScrollbarWidth); |
| } |
| |
| int physicalScrollX = scrollPosition().x() + scrollOrigin().x(); |
| if (physicalScrollX < 0) { |
| verticalOverhangRect.setWidth(-physicalScrollX); |
| verticalOverhangRect.setHeight(frameRect().height() - horizontalOverhangRect.height() - horizontalScrollbarHeight); |
| verticalOverhangRect.setX(frameRect().x()); |
| if (horizontalOverhangRect.y() == frameRect().y()) |
| verticalOverhangRect.setY(frameRect().y() + horizontalOverhangRect.height()); |
| else |
| verticalOverhangRect.setY(frameRect().y()); |
| } else if (contentsWidth() && physicalScrollX > contentsWidth() - visibleWidth()) { |
| int width = physicalScrollX - (contentsWidth() - visibleWidth()); |
| verticalOverhangRect.setWidth(width); |
| verticalOverhangRect.setHeight(frameRect().height() - horizontalOverhangRect.height() - horizontalScrollbarHeight); |
| verticalOverhangRect.setX(frameRect().maxX() - width - verticalScrollbarWidth); |
| if (horizontalOverhangRect.y() == frameRect().y()) |
| verticalOverhangRect.setY(frameRect().y() + horizontalOverhangRect.height()); |
| else |
| verticalOverhangRect.setY(frameRect().y()); |
| } |
| } |
| |
| void ScrollView::updateOverhangAreas() |
| { |
| HostWindow* window = hostWindow(); |
| if (!window) |
| return; |
| |
| IntRect horizontalOverhangRect; |
| IntRect verticalOverhangRect; |
| calculateOverhangAreasForPainting(horizontalOverhangRect, verticalOverhangRect); |
| if (!horizontalOverhangRect.isEmpty()) |
| window->invalidateContentsAndRootView(horizontalOverhangRect); |
| if (!verticalOverhangRect.isEmpty()) |
| window->invalidateContentsAndRootView(verticalOverhangRect); |
| } |
| |
| void ScrollView::paintOverhangAreas(GraphicsContext* context, const IntRect& horizontalOverhangRect, const IntRect& verticalOverhangRect, const IntRect& dirtyRect) |
| { |
| ScrollbarTheme::theme()->paintOverhangAreas(this, context, horizontalOverhangRect, verticalOverhangRect, dirtyRect); |
| } |
| |
| void ScrollView::calculateAndPaintOverhangAreas(GraphicsContext* context, const IntRect& dirtyRect) |
| { |
| IntRect horizontalOverhangRect; |
| IntRect verticalOverhangRect; |
| calculateOverhangAreasForPainting(horizontalOverhangRect, verticalOverhangRect); |
| |
| if (dirtyRect.intersects(horizontalOverhangRect) || dirtyRect.intersects(verticalOverhangRect)) |
| paintOverhangAreas(context, horizontalOverhangRect, verticalOverhangRect, dirtyRect); |
| } |
| |
| bool ScrollView::isPointInScrollbarCorner(const IntPoint& windowPoint) |
| { |
| if (!scrollbarCornerPresent()) |
| return false; |
| |
| IntPoint viewPoint = convertFromContainingWindow(windowPoint); |
| |
| if (m_horizontalScrollbar) { |
| int horizontalScrollbarYMin = m_horizontalScrollbar->frameRect().y(); |
| int horizontalScrollbarYMax = m_horizontalScrollbar->frameRect().y() + m_horizontalScrollbar->frameRect().height(); |
| int horizontalScrollbarXMin = m_horizontalScrollbar->frameRect().x() + m_horizontalScrollbar->frameRect().width(); |
| |
| return viewPoint.y() > horizontalScrollbarYMin && viewPoint.y() < horizontalScrollbarYMax && viewPoint.x() > horizontalScrollbarXMin; |
| } |
| |
| int verticalScrollbarXMin = m_verticalScrollbar->frameRect().x(); |
| int verticalScrollbarXMax = m_verticalScrollbar->frameRect().x() + m_verticalScrollbar->frameRect().width(); |
| int verticalScrollbarYMin = m_verticalScrollbar->frameRect().y() + m_verticalScrollbar->frameRect().height(); |
| |
| return viewPoint.x() > verticalScrollbarXMin && viewPoint.x() < verticalScrollbarXMax && viewPoint.y() > verticalScrollbarYMin; |
| } |
| |
| bool ScrollView::scrollbarCornerPresent() const |
| { |
| return (m_horizontalScrollbar && width() - m_horizontalScrollbar->width() > 0) |
| || (m_verticalScrollbar && height() - m_verticalScrollbar->height() > 0); |
| } |
| |
| IntRect ScrollView::convertFromScrollbarToContainingView(const Scrollbar* scrollbar, const IntRect& localRect) const |
| { |
| // Scrollbars won't be transformed within us |
| IntRect newRect = localRect; |
| newRect.moveBy(scrollbar->location()); |
| return newRect; |
| } |
| |
| IntRect ScrollView::convertFromContainingViewToScrollbar(const Scrollbar* scrollbar, const IntRect& parentRect) const |
| { |
| IntRect newRect = parentRect; |
| // Scrollbars won't be transformed within us |
| newRect.moveBy(-scrollbar->location()); |
| return newRect; |
| } |
| |
| // FIXME: test these on windows |
| IntPoint ScrollView::convertFromScrollbarToContainingView(const Scrollbar* scrollbar, const IntPoint& localPoint) const |
| { |
| // Scrollbars won't be transformed within us |
| IntPoint newPoint = localPoint; |
| newPoint.moveBy(scrollbar->location()); |
| return newPoint; |
| } |
| |
| IntPoint ScrollView::convertFromContainingViewToScrollbar(const Scrollbar* scrollbar, const IntPoint& parentPoint) const |
| { |
| IntPoint newPoint = parentPoint; |
| // Scrollbars won't be transformed within us |
| newPoint.moveBy(-scrollbar->location()); |
| return newPoint; |
| } |
| |
| void ScrollView::setParentVisible(bool visible) |
| { |
| if (isParentVisible() == visible) |
| return; |
| |
| Widget::setParentVisible(visible); |
| |
| if (!isSelfVisible()) |
| return; |
| |
| HashSet<RefPtr<Widget>>::iterator end = m_children.end(); |
| for (HashSet<RefPtr<Widget>>::iterator it = m_children.begin(); it != end; ++it) |
| (*it)->setParentVisible(visible); |
| } |
| |
| void ScrollView::show() |
| { |
| if (!isSelfVisible()) { |
| setSelfVisible(true); |
| if (isParentVisible()) { |
| HashSet<RefPtr<Widget>>::iterator end = m_children.end(); |
| for (HashSet<RefPtr<Widget>>::iterator it = m_children.begin(); it != end; ++it) |
| (*it)->setParentVisible(true); |
| } |
| } |
| |
| Widget::show(); |
| } |
| |
| void ScrollView::hide() |
| { |
| if (isSelfVisible()) { |
| if (isParentVisible()) { |
| HashSet<RefPtr<Widget>>::iterator end = m_children.end(); |
| for (HashSet<RefPtr<Widget>>::iterator it = m_children.begin(); it != end; ++it) |
| (*it)->setParentVisible(false); |
| } |
| setSelfVisible(false); |
| } |
| |
| Widget::hide(); |
| } |
| |
| bool ScrollView::isOffscreen() const |
| { |
| if (platformWidget()) |
| return platformIsOffscreen(); |
| |
| if (!isVisible()) |
| return true; |
| |
| // FIXME: Add a HostWindow::isOffscreen method here. Since only Mac implements this method |
| // currently, we can add the method when the other platforms decide to implement this concept. |
| return false; |
| } |
| |
| |
| void ScrollView::addPanScrollIcon(const IntPoint& iconPosition) |
| { |
| HostWindow* window = hostWindow(); |
| if (!window) |
| return; |
| m_drawPanScrollIcon = true; |
| m_panScrollIconPoint = IntPoint(iconPosition.x() - panIconSizeLength / 2 , iconPosition.y() - panIconSizeLength / 2) ; |
| window->invalidateContentsAndRootView(IntRect(m_panScrollIconPoint, IntSize(panIconSizeLength, panIconSizeLength))); |
| } |
| |
| void ScrollView::removePanScrollIcon() |
| { |
| HostWindow* window = hostWindow(); |
| if (!window) |
| return; |
| m_drawPanScrollIcon = false; |
| window->invalidateContentsAndRootView(IntRect(m_panScrollIconPoint, IntSize(panIconSizeLength, panIconSizeLength))); |
| } |
| |
| void ScrollView::setScrollOrigin(const IntPoint& origin, bool updatePositionAtAll, bool updatePositionSynchronously) |
| { |
| if (scrollOrigin() == origin) |
| return; |
| |
| ScrollableArea::setScrollOrigin(origin); |
| |
| if (platformWidget()) { |
| platformSetScrollOrigin(origin, updatePositionAtAll, updatePositionSynchronously); |
| return; |
| } |
| |
| // Update if the scroll origin changes, since our position will be different if the content size did not change. |
| if (updatePositionAtAll && updatePositionSynchronously) |
| updateScrollbars(scrollOffset()); |
| } |
| |
| void ScrollView::styleDidChange() |
| { |
| if (m_horizontalScrollbar) |
| m_horizontalScrollbar->styleChanged(); |
| |
| if (m_verticalScrollbar) |
| m_verticalScrollbar->styleChanged(); |
| } |
| |
| #if !PLATFORM(COCOA) |
| |
| void ScrollView::platformAddChild(Widget*) |
| { |
| } |
| |
| void ScrollView::platformRemoveChild(Widget*) |
| { |
| } |
| |
| #endif |
| |
| #if !PLATFORM(COCOA) |
| |
| void ScrollView::platformSetScrollbarsSuppressed(bool) |
| { |
| } |
| |
| void ScrollView::platformSetScrollOrigin(const IntPoint&, bool, bool) |
| { |
| } |
| |
| void ScrollView::platformSetScrollbarOverlayStyle(ScrollbarOverlayStyle) |
| { |
| } |
| |
| #endif |
| |
| #if !PLATFORM(COCOA) |
| |
| void ScrollView::platformSetScrollbarModes() |
| { |
| } |
| |
| void ScrollView::platformScrollbarModes(ScrollbarMode& horizontal, ScrollbarMode& vertical) const |
| { |
| horizontal = ScrollbarAuto; |
| vertical = ScrollbarAuto; |
| } |
| |
| void ScrollView::platformSetCanBlitOnScroll(bool) |
| { |
| } |
| |
| bool ScrollView::platformCanBlitOnScroll() const |
| { |
| return false; |
| } |
| |
| IntRect ScrollView::platformVisibleContentRect(bool) const |
| { |
| return IntRect(); |
| } |
| |
| float ScrollView::platformTopContentInset() const |
| { |
| return 0; |
| } |
| |
| void ScrollView::platformSetTopContentInset(float) |
| { |
| } |
| |
| IntSize ScrollView::platformVisibleContentSize(bool) const |
| { |
| return IntSize(); |
| } |
| |
| IntRect ScrollView::platformVisibleContentRectIncludingObscuredArea(bool) const |
| { |
| return IntRect(); |
| } |
| |
| IntSize ScrollView::platformVisibleContentSizeIncludingObscuredArea(bool) const |
| { |
| return IntSize(); |
| } |
| |
| void ScrollView::platformSetContentsSize() |
| { |
| } |
| |
| IntRect ScrollView::platformContentsToScreen(const IntRect& rect) const |
| { |
| return rect; |
| } |
| |
| IntPoint ScrollView::platformScreenToContents(const IntPoint& point) const |
| { |
| return point; |
| } |
| |
| void ScrollView::platformSetScrollPosition(const IntPoint&) |
| { |
| } |
| |
| bool ScrollView::platformScroll(ScrollDirection, ScrollGranularity) |
| { |
| return true; |
| } |
| |
| void ScrollView::platformRepaintContentRectangle(const IntRect&) |
| { |
| } |
| |
| bool ScrollView::platformIsOffscreen() const |
| { |
| return false; |
| } |
| |
| #endif |
| |
| } |