/* * Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies) * Copyright (C) 2009 Antonio Gomes <tonikitoo@webkit.org> * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. */ #ifndef SpatialNavigation_h #define SpatialNavigation_h #include "FocusDirection.h" #include "HTMLFrameOwnerElement.h" #include "IntRect.h" #include "Node.h" #include <limits> namespace WebCore { class Element; class Frame; class HTMLAreaElement; class IntRect; class RenderObject; using namespace std; inline long long maxDistance() { return numeric_limits<long long>::max(); } inline int fudgeFactor() { return 2; } bool isSpatialNavigationEnabled(const Frame*); // Spatially speaking, two given elements in a web page can be: // 1) Fully aligned: There is a full intersection between the rects, either // vertically or horizontally. // // * Horizontally * Vertically // _ // |_| _ _ _ _ _ _ // |_|...... _ |_|_|_|_|_|_| // |_| |_| . . // |_|......|_| OR . . // |_| |_| . . // |_|......|_| _ _ _ _ // |_| |_|_|_|_| // // // 2) Partially aligned: There is a partial intersection between the rects, either // vertically or horizontally. // // * Horizontally * Vertically // _ _ _ _ _ _ // |_| |_|_|_|_|_| // |_|.... _ OR . . // |_| |_| . . // |_|....|_| ._._ _ // |_| |_|_|_| // |_| // // 3) Or, otherwise, not aligned at all. // // * Horizontally * Vertically // _ _ _ _ _ // |_| |_|_|_|_| // |_| . // |_| . // . OR . // _ . ._ _ _ _ _ // |_| |_|_|_|_|_| // |_| // |_| // // "Totally Aligned" elements are preferable candidates to move // focus to over "Partially Aligned" ones, that on its turns are // more preferable than "Not Aligned". enum RectsAlignment { None = 0, Partial, Full }; struct FocusCandidate { FocusCandidate() : visibleNode(0) , focusableNode(0) , enclosingScrollableBox(0) , distance(maxDistance()) , parentDistance(maxDistance()) , alignment(None) , parentAlignment(None) , isOffscreen(true) , isOffscreenAfterScrolling(true) { } FocusCandidate(Node* n, FocusDirection); explicit FocusCandidate(HTMLAreaElement* area, FocusDirection); bool isNull() const { return !visibleNode; } bool inScrollableContainer() const { return visibleNode && enclosingScrollableBox; } bool isFrameOwnerElement() const { return visibleNode && visibleNode->isFrameOwnerElement(); } Document* document() const { return visibleNode ? visibleNode->document() : 0; } // We handle differently visibleNode and FocusableNode to properly handle the areas of imagemaps, // where visibleNode would represent the image element and focusableNode would represent the area element. // In all other cases, visibleNode and focusableNode are one and the same. Node* visibleNode; Node* focusableNode; Node* enclosingScrollableBox; long long distance; long long parentDistance; RectsAlignment alignment; RectsAlignment parentAlignment; IntRect rect; bool isOffscreen; bool isOffscreenAfterScrolling; }; bool hasOffscreenRect(Node*, FocusDirection direction = FocusDirectionNone); bool scrollInDirection(Frame*, FocusDirection); bool scrollInDirection(Node* container, FocusDirection); bool canScrollInDirection(const Node* container, FocusDirection); bool canScrollInDirection(const Frame*, FocusDirection); bool canBeScrolledIntoView(FocusDirection, const FocusCandidate&); bool areElementsOnSameLine(const FocusCandidate& firstCandidate, const FocusCandidate& secondCandidate); void distanceDataForNode(FocusDirection, const FocusCandidate& current, FocusCandidate& candidate); Node* scrollableEnclosingBoxOrParentFrameForNodeInDirection(FocusDirection, Node*); IntRect nodeRectInAbsoluteCoordinates(Node*, bool ignoreBorder = false); IntRect frameRectInAbsoluteCoordinates(Frame*); IntRect virtualRectForDirection(FocusDirection, const IntRect& startingRect, int width = 0); IntRect virtualRectForAreaElementAndDirection(HTMLAreaElement*, FocusDirection); HTMLFrameOwnerElement* frameOwnerElement(FocusCandidate&); } // namspace WebCore #endif // SpatialNavigation_h