123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447 |
- /*
- * Copyright (c) 2010, Google Inc. All rights reserved.
- * Copyright (C) 2008, 2011 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:
- *
- * * Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * * 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.
- * * Neither the name of Google Inc. nor the names of its
- * contributors may be used to endorse or promote products derived from
- * this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * "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 THE COPYRIGHT
- * OWNER 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 "ScrollableArea.h"
- #include "GraphicsContext.h"
- #include "GraphicsLayer.h"
- #include "FloatPoint.h"
- #include "PlatformWheelEvent.h"
- #include "ScrollAnimator.h"
- #include "ScrollbarTheme.h"
- #include <wtf/PassOwnPtr.h>
- namespace WebCore {
- struct SameSizeAsScrollableArea {
- virtual ~SameSizeAsScrollableArea();
- void* pointer;
- unsigned bitfields : 16;
- IntPoint origin;
- };
- COMPILE_ASSERT(sizeof(ScrollableArea) == sizeof(SameSizeAsScrollableArea), ScrollableArea_should_stay_small);
- ScrollableArea::ScrollableArea()
- : m_constrainsScrollingToContentEdge(true)
- , m_inLiveResize(false)
- , m_verticalScrollElasticity(ScrollElasticityNone)
- , m_horizontalScrollElasticity(ScrollElasticityNone)
- , m_scrollbarOverlayStyle(ScrollbarOverlayStyleDefault)
- , m_scrollOriginChanged(false)
- {
- }
- ScrollableArea::~ScrollableArea()
- {
- }
- ScrollAnimator* ScrollableArea::scrollAnimator() const
- {
- if (!m_scrollAnimator)
- m_scrollAnimator = ScrollAnimator::create(const_cast<ScrollableArea*>(this));
- return m_scrollAnimator.get();
- }
- void ScrollableArea::setScrollOrigin(const IntPoint& origin)
- {
- if (m_scrollOrigin != origin) {
- m_scrollOrigin = origin;
- m_scrollOriginChanged = true;
- }
- }
- bool ScrollableArea::scroll(ScrollDirection direction, ScrollGranularity granularity, float multiplier)
- {
- ScrollbarOrientation orientation;
- Scrollbar* scrollbar;
- if (direction == ScrollUp || direction == ScrollDown) {
- orientation = VerticalScrollbar;
- scrollbar = verticalScrollbar();
- } else {
- orientation = HorizontalScrollbar;
- scrollbar = horizontalScrollbar();
- }
- if (!scrollbar)
- return false;
- float step = 0;
- switch (granularity) {
- case ScrollByLine:
- step = scrollbar->lineStep();
- break;
- case ScrollByPage:
- step = scrollbar->pageStep();
- break;
- case ScrollByDocument:
- step = scrollbar->totalSize();
- break;
- case ScrollByPixel:
- case ScrollByPrecisePixel:
- step = scrollbar->pixelStep();
- break;
- }
- if (direction == ScrollUp || direction == ScrollLeft)
- multiplier = -multiplier;
- return scrollAnimator()->scroll(orientation, granularity, step, multiplier);
- }
- void ScrollableArea::scrollToOffsetWithoutAnimation(const FloatPoint& offset)
- {
- scrollAnimator()->scrollToOffsetWithoutAnimation(offset);
- }
- void ScrollableArea::scrollToOffsetWithoutAnimation(ScrollbarOrientation orientation, float offset)
- {
- if (orientation == HorizontalScrollbar)
- scrollToOffsetWithoutAnimation(FloatPoint(offset, scrollAnimator()->currentPosition().y()));
- else
- scrollToOffsetWithoutAnimation(FloatPoint(scrollAnimator()->currentPosition().x(), offset));
- }
- void ScrollableArea::notifyScrollPositionChanged(const IntPoint& position)
- {
- scrollPositionChanged(position);
- scrollAnimator()->setCurrentPosition(position);
- }
- void ScrollableArea::scrollPositionChanged(const IntPoint& position)
- {
- IntPoint oldPosition = scrollPosition();
- // Tell the derived class to scroll its contents.
- setScrollOffset(position);
- Scrollbar* verticalScrollbar = this->verticalScrollbar();
- // Tell the scrollbars to update their thumb postions.
- if (Scrollbar* horizontalScrollbar = this->horizontalScrollbar()) {
- horizontalScrollbar->offsetDidChange();
- if (horizontalScrollbar->isOverlayScrollbar() && !hasLayerForHorizontalScrollbar()) {
- if (!verticalScrollbar)
- horizontalScrollbar->invalidate();
- else {
- // If there is both a horizontalScrollbar and a verticalScrollbar,
- // then we must also invalidate the corner between them.
- IntRect boundsAndCorner = horizontalScrollbar->boundsRect();
- boundsAndCorner.setWidth(boundsAndCorner.width() + verticalScrollbar->width());
- horizontalScrollbar->invalidateRect(boundsAndCorner);
- }
- }
- }
- if (verticalScrollbar) {
- verticalScrollbar->offsetDidChange();
- if (verticalScrollbar->isOverlayScrollbar() && !hasLayerForVerticalScrollbar())
- verticalScrollbar->invalidate();
- }
- if (scrollPosition() != oldPosition)
- scrollAnimator()->notifyContentAreaScrolled(scrollPosition() - oldPosition);
- }
- bool ScrollableArea::handleWheelEvent(const PlatformWheelEvent& wheelEvent)
- {
- return scrollAnimator()->handleWheelEvent(wheelEvent);
- }
- // NOTE: Only called from Internals for testing.
- void ScrollableArea::setScrollOffsetFromInternals(const IntPoint& offset)
- {
- setScrollOffsetFromAnimation(offset);
- }
- void ScrollableArea::setScrollOffsetFromAnimation(const IntPoint& offset)
- {
- if (requestScrollPositionUpdate(offset))
- return;
- scrollPositionChanged(offset);
- }
- void ScrollableArea::willStartLiveResize()
- {
- if (m_inLiveResize)
- return;
- m_inLiveResize = true;
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->willStartLiveResize();
- }
- void ScrollableArea::willEndLiveResize()
- {
- if (!m_inLiveResize)
- return;
- m_inLiveResize = false;
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->willEndLiveResize();
- }
- void ScrollableArea::contentAreaWillPaint() const
- {
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->contentAreaWillPaint();
- }
- void ScrollableArea::mouseEnteredContentArea() const
- {
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->mouseEnteredContentArea();
- }
- void ScrollableArea::mouseExitedContentArea() const
- {
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->mouseEnteredContentArea();
- }
- void ScrollableArea::mouseMovedInContentArea() const
- {
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->mouseMovedInContentArea();
- }
- void ScrollableArea::mouseEnteredScrollbar(Scrollbar* scrollbar) const
- {
- scrollAnimator()->mouseEnteredScrollbar(scrollbar);
- }
- void ScrollableArea::mouseExitedScrollbar(Scrollbar* scrollbar) const
- {
- scrollAnimator()->mouseExitedScrollbar(scrollbar);
- }
- void ScrollableArea::contentAreaDidShow() const
- {
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->contentAreaDidShow();
- }
- void ScrollableArea::contentAreaDidHide() const
- {
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->contentAreaDidHide();
- }
- void ScrollableArea::finishCurrentScrollAnimations() const
- {
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->finishCurrentScrollAnimations();
- }
- void ScrollableArea::didAddScrollbar(Scrollbar* scrollbar, ScrollbarOrientation orientation)
- {
- if (orientation == VerticalScrollbar)
- scrollAnimator()->didAddVerticalScrollbar(scrollbar);
- else
- scrollAnimator()->didAddHorizontalScrollbar(scrollbar);
- // <rdar://problem/9797253> AppKit resets the scrollbar's style when you attach a scrollbar
- setScrollbarOverlayStyle(scrollbarOverlayStyle());
- }
- void ScrollableArea::willRemoveScrollbar(Scrollbar* scrollbar, ScrollbarOrientation orientation)
- {
- if (orientation == VerticalScrollbar)
- scrollAnimator()->willRemoveVerticalScrollbar(scrollbar);
- else
- scrollAnimator()->willRemoveHorizontalScrollbar(scrollbar);
- }
- void ScrollableArea::contentsResized()
- {
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->contentsResized();
- }
- bool ScrollableArea::hasOverlayScrollbars() const
- {
- return (verticalScrollbar() && verticalScrollbar()->isOverlayScrollbar())
- || (horizontalScrollbar() && horizontalScrollbar()->isOverlayScrollbar());
- }
- void ScrollableArea::setScrollbarOverlayStyle(ScrollbarOverlayStyle overlayStyle)
- {
- m_scrollbarOverlayStyle = overlayStyle;
- if (horizontalScrollbar()) {
- ScrollbarTheme::theme()->updateScrollbarOverlayStyle(horizontalScrollbar());
- horizontalScrollbar()->invalidate();
- }
-
- if (verticalScrollbar()) {
- ScrollbarTheme::theme()->updateScrollbarOverlayStyle(verticalScrollbar());
- verticalScrollbar()->invalidate();
- }
- }
- void ScrollableArea::invalidateScrollbar(Scrollbar* scrollbar, const IntRect& rect)
- {
- #if USE(ACCELERATED_COMPOSITING)
- if (scrollbar == horizontalScrollbar()) {
- if (GraphicsLayer* graphicsLayer = layerForHorizontalScrollbar()) {
- graphicsLayer->setNeedsDisplay();
- graphicsLayer->setContentsNeedsDisplay();
- return;
- }
- } else if (scrollbar == verticalScrollbar()) {
- if (GraphicsLayer* graphicsLayer = layerForVerticalScrollbar()) {
- graphicsLayer->setNeedsDisplay();
- graphicsLayer->setContentsNeedsDisplay();
- return;
- }
- }
- #endif
- invalidateScrollbarRect(scrollbar, rect);
- }
- void ScrollableArea::invalidateScrollCorner(const IntRect& rect)
- {
- #if USE(ACCELERATED_COMPOSITING)
- if (GraphicsLayer* graphicsLayer = layerForScrollCorner()) {
- graphicsLayer->setNeedsDisplay();
- return;
- }
- #endif
- invalidateScrollCornerRect(rect);
- }
- bool ScrollableArea::hasLayerForHorizontalScrollbar() const
- {
- #if USE(ACCELERATED_COMPOSITING)
- return layerForHorizontalScrollbar();
- #else
- return false;
- #endif
- }
- bool ScrollableArea::hasLayerForVerticalScrollbar() const
- {
- #if USE(ACCELERATED_COMPOSITING)
- return layerForVerticalScrollbar();
- #else
- return false;
- #endif
- }
- bool ScrollableArea::hasLayerForScrollCorner() const
- {
- #if USE(ACCELERATED_COMPOSITING)
- return layerForScrollCorner();
- #else
- return false;
- #endif
- }
- void ScrollableArea::serviceScrollAnimations()
- {
- if (ScrollAnimator* scrollAnimator = existingScrollAnimator())
- scrollAnimator->serviceScrollAnimations();
- }
- IntPoint ScrollableArea::scrollPosition() const
- {
- int x = horizontalScrollbar() ? horizontalScrollbar()->value() : 0;
- int y = verticalScrollbar() ? verticalScrollbar()->value() : 0;
- return IntPoint(x, y);
- }
- IntPoint ScrollableArea::minimumScrollPosition() const
- {
- return IntPoint();
- }
- IntPoint ScrollableArea::maximumScrollPosition() const
- {
- return IntPoint(totalContentsSize().width() - visibleWidth(), totalContentsSize().height() - visibleHeight());
- }
- IntSize ScrollableArea::totalContentsSize() const
- {
- IntSize totalContentsSize = contentsSize();
- totalContentsSize.setHeight(totalContentsSize.height() + headerHeight() + footerHeight());
- return totalContentsSize;
- }
- IntRect ScrollableArea::visibleContentRect(VisibleContentRectIncludesScrollbars scrollbarInclusion) const
- {
- int verticalScrollbarWidth = 0;
- int horizontalScrollbarHeight = 0;
- if (scrollbarInclusion == IncludeScrollbars) {
- if (Scrollbar* verticalBar = verticalScrollbar())
- verticalScrollbarWidth = !verticalBar->isOverlayScrollbar() ? verticalBar->width() : 0;
- if (Scrollbar* horizontalBar = horizontalScrollbar())
- horizontalScrollbarHeight = !horizontalBar->isOverlayScrollbar() ? horizontalBar->height() : 0;
- }
- return IntRect(scrollPosition().x(),
- scrollPosition().y(),
- std::max(0, visibleWidth() + verticalScrollbarWidth),
- std::max(0, visibleHeight() + horizontalScrollbarHeight));
- }
- IntPoint ScrollableArea::constrainScrollPositionForOverhang(const IntRect& visibleContentRect, const IntSize& totalContentsSize, const IntPoint& scrollPosition, const IntPoint& scrollOrigin, int headerHeight, int footerHeight)
- {
- // The viewport rect that we're scrolling shouldn't be larger than our document.
- IntSize idealScrollRectSize(std::min(visibleContentRect.width(), totalContentsSize.width()), std::min(visibleContentRect.height(), totalContentsSize.height()));
-
- IntRect scrollRect(scrollPosition + scrollOrigin - IntSize(0, headerHeight), idealScrollRectSize);
- IntRect documentRect(IntPoint(), IntSize(totalContentsSize.width(), totalContentsSize.height() - headerHeight - footerHeight));
- // Use intersection to constrain our ideal scroll rect by the document rect.
- scrollRect.intersect(documentRect);
- if (scrollRect.size() != idealScrollRectSize) {
- // If the rect was clipped, restore its size, effectively pushing it "down" from the top left.
- scrollRect.setSize(idealScrollRectSize);
- // If we still clip, push our rect "up" from the bottom right.
- scrollRect.intersect(documentRect);
- if (scrollRect.width() < idealScrollRectSize.width())
- scrollRect.move(-(idealScrollRectSize.width() - scrollRect.width()), 0);
- if (scrollRect.height() < idealScrollRectSize.height())
- scrollRect.move(0, -(idealScrollRectSize.height() - scrollRect.height()));
- }
- return scrollRect.location() - toIntSize(scrollOrigin);
- }
- IntPoint ScrollableArea::constrainScrollPositionForOverhang(const IntPoint& scrollPosition)
- {
- return constrainScrollPositionForOverhang(visibleContentRect(), totalContentsSize(), scrollPosition, scrollOrigin(), headerHeight(), footerHeight());
- }
- } // namespace WebCore
|