| Index: Source/core/html/canvas/CanvasHitRegion.cpp | 
| diff --git a/Source/core/html/canvas/CanvasHitRegion.cpp b/Source/core/html/canvas/CanvasHitRegion.cpp | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..0841a4128d5763ec27b8f0133e9ff27737f57196 | 
| --- /dev/null | 
| +++ b/Source/core/html/canvas/CanvasHitRegion.cpp | 
| @@ -0,0 +1,378 @@ | 
| +// Copyright 2014 The Chromium Authors. All rights reserved. | 
| +// Use of this source code is governed by a BSD-style license that can be | 
| +// found in the LICENSE file. | 
| + | 
| +#include "config.h" | 
| +#include "core/html/canvas/CanvasHitRegion.h" | 
| + | 
| +#include "bindings/v8/Dictionary.h" | 
| +#include "bindings/v8/ExceptionState.h" | 
| +#include "core/dom/Element.h" | 
| +#include "core/dom/ElementTraversal.h" | 
| +#include "core/html/HTMLImageElement.h" | 
| +#include "core/html/HTMLInputElement.h" | 
| +#include "core/html/HTMLSelectElement.h" | 
| +#include "core/html/canvas/Path2D.h" | 
| + | 
| +namespace WebCore { | 
| + | 
| +// FIXME: De-dupe. | 
| +static WindRule parseWinding(const String& windingRuleString, WindRule defaultValue = RULE_EVENODD) | 
| +{ | 
| +    if (windingRuleString == "nonzero") | 
| +        return RULE_NONZERO; | 
| +    if (windingRuleString == "evenodd") | 
| +        return RULE_EVENODD; | 
| + | 
| +    return defaultValue; | 
| +} | 
| + | 
| +DecodedHitRegionOptions::DecodedHitRegionOptions() | 
| +    : hasPath(false) | 
| +    , fillRule(RULE_NONZERO) | 
| +{ | 
| +} | 
| + | 
| +DecodedHitRegionOptions::DecodedHitRegionOptions(const Dictionary& options) | 
| +    : hasPath(false) | 
| +    , fillRule(RULE_NONZERO) | 
| +{ | 
| +    RefPtr<Path2D> pathFromDictionary; | 
| +    if (options.get("path", pathFromDictionary) && pathFromDictionary) { | 
| +        path = pathFromDictionary->path(); | 
| +        hasPath = true; | 
| +    } | 
| +    String windRuleString; | 
| +    if (options.get("fillRule", windRuleString)) | 
| +        fillRule = parseWinding(windRuleString, RULE_NONZERO); | 
| +    if (options.get("id", id) && id.isEmpty()) | 
| +        id = String(); | 
| +    if (options.getWithUndefinedOrNullCheck("parentID", parentId) && parentId.isEmpty()) | 
| +        parentId = String(); | 
| +    options.get("cursor", cursor); | 
| +    options.get("control", control); | 
| +    if (options.getWithUndefinedOrNullCheck("label", label) && label.isEmpty()) | 
| +        label = String(); | 
| +    if (options.getWithUndefinedOrNullCheck("role", role) && role.isEmpty()) | 
| +        role = String(); | 
| +} | 
| + | 
| +void DecodedHitRegionOptions::resolvePath(const Path& currentPath, const AffineTransform& currentTransform) | 
| +{ | 
| +    if (!hasPath) | 
| +        path = currentPath; | 
| +    path.transform(currentTransform); | 
| +} | 
| + | 
| +void DecodedHitRegionOptions::resolveIds(const CanvasHitRegionManager* hitRegionManager) | 
| +{ | 
| +    parentHitRegion = !parentId.isEmpty() ? hitRegionManager->getHitRegionById(parentId) : 0; | 
| +    previousHitRegion = !id.isEmpty() ? hitRegionManager->getHitRegionById(id) : 0; | 
| +} | 
| + | 
| +static bool isSupportedInteractiveFallbackElement(Element& element) | 
| +{ | 
| +    // "An element is a supported interactive canvas fallback element if it is one of the following:" | 
| + | 
| +    // "an 'a' element that represents a hyperlink and that does not have any img descendants" | 
| +    if (element.hasTagName(HTMLNames::aTag)) | 
| +        return !Traversal<HTMLImageElement>::firstWithin(element); | 
| + | 
| +    // "a button element" | 
| +    if (element.hasTagName(HTMLNames::buttonTag)) | 
| +        return true; | 
| + | 
| +    if (isHTMLInputElement(element)) { | 
| +        // "an input element whose type attribute is in one of the Checkbox or Radio Button states" | 
| +        const HTMLInputElement& inputElement = toHTMLInputElement(element); | 
| +        if (inputElement.isCheckbox() || inputElement.isRadioButton()) | 
| +            return true; | 
| + | 
| +        // "an input element that is a button but its type attribute is not in the Image Button state" | 
| +        if (inputElement.isTextButton() && !inputElement.isImageButton()) | 
| +            return true; | 
| +    } | 
| + | 
| +    // "a select element with a multiple attribute or a display size greater than 1" | 
| +    if (isHTMLSelectElement(element)) { | 
| +        const HTMLSelectElement& selectElement = toHTMLSelectElement(element); | 
| +        if (selectElement.multiple() || selectElement.size() > 1) | 
| +            return true; | 
| +    } | 
| + | 
| +    // "an option element that is in a list of options of a select element with | 
| +    // a multiple attribute or a display size greater than 1" | 
| +    // FIXME: Only direct descendant? | 
| +    if (isHTMLOptionElement(element) && element.parentNode() && isHTMLSelectElement(*element.parentNode())) { | 
| +        const HTMLSelectElement& selectElement = toHTMLSelectElement(*element.parentNode()); | 
| +        if (selectElement.multiple() || selectElement.size() > 1) | 
| +            return true; | 
| +    } | 
| + | 
| +    // "a sorting interface th element" | 
| +    // Note: This seems redundant with the last condition. | 
| +    if (element.hasTagName(HTMLNames::thTag) && element.fastHasAttribute(HTMLNames::sortableAttr)) | 
| +        return true; | 
| + | 
| +    // "an element that would not be interactive content except for having the | 
| +    // tabindex attribute specified" | 
| +    // FIXME: Does not test the "interactive content" part. | 
| +    if (element.fastHasAttribute(HTMLNames::tabindexAttr)) | 
| +        return true; | 
| + | 
| +    // "a non-interactive table, caption, thead, tbody, tfoot, tr, td, or th element" | 
| +    if (element.hasTagName(HTMLNames::tableTag) | 
| +        || element.hasTagName(HTMLNames::captionTag) | 
| +        || element.hasTagName(HTMLNames::theadTag) | 
| +        || element.hasTagName(HTMLNames::tbodyTag) | 
| +        || element.hasTagName(HTMLNames::tfootTag) | 
| +        || element.hasTagName(HTMLNames::trTag) | 
| +        || element.hasTagName(HTMLNames::tdTag) | 
| +        || element.hasTagName(HTMLNames::thTag)) | 
| +        return true; | 
| + | 
| +    return false; | 
| +} | 
| + | 
| +static bool isValidCSSCursor(const String& cursor) | 
| +{ | 
| +    // FIXME: Implement. | 
| +    return false; | 
| +} | 
| + | 
| +static bool isValidCursorValue(const String& cursor) | 
| +{ | 
| +    return cursor == "inherit" || isValidCSSCursor(cursor); | 
| +} | 
| + | 
| +static bool hasValidAriaRoles(const String& roleList) | 
| +{ | 
| +    // FIXME: Implement. | 
| +    return false; | 
| +} | 
| + | 
| +bool DecodedHitRegionOptions::validate(ExceptionState& exceptionState) const | 
| +{ | 
| +    // 11. "If any of the following conditions are met, throw a NotSupportedError exception and abort these steps." | 
| + | 
| +    // "The arguments object's control and label members are both non-null." | 
| +    if (control && !label.isNull()) | 
| +        exceptionState.throwDOMException(NotSupportedError, "cannot specify both a control and a label"); | 
| + | 
| +    // "The arguments object's control and role members are both non-null." | 
| +    if (control && !role.isNull()) | 
| +        exceptionState.throwDOMException(NotSupportedError, "cannot specify both a control and a role"); | 
| + | 
| +    // "The arguments object's role member's value is the empty string, and the | 
| +    // label member's value is either null or the empty string." | 
| +    if (role.isEmpty() && (label.isNull() || label.isEmpty())) | 
| +        exceptionState.throwDOMException(NotSupportedError, "both role and label cannot be empty"); | 
| + | 
| +    // FIXME: Include clip. | 
| +    // "The specified pixels has no pixels." (s/pixels/area/) | 
| +    if (path.isEmpty() || path.boundingRect().isEmpty()) | 
| +        exceptionState.throwDOMException(NotSupportedError, "the specified region cannot be empty"); | 
| + | 
| +    // "The arguments object's control member is neither null nor a supported | 
| +    // interactive canvas fallback element." | 
| +    if (control && !isSupportedInteractiveFallbackElement(*control)) | 
| +        exceptionState.throwDOMException(NotSupportedError, "the specified control is not supported"); | 
| + | 
| +    // "The parent region is not null but has a control." | 
| +    if (parentHitRegion && parentHitRegion->control()) | 
| +        exceptionState.throwDOMException(NotSupportedError, "the specified parent hit-region is associated with a control"); | 
| + | 
| +    // "The previous region for this ID is the same hit region as the parent region." | 
| +    if (previousHitRegion && parentHitRegion && previousHitRegion == parentHitRegion) | 
| +        exceptionState.throwDOMException(NotSupportedError, "the previous region is the same as the parent region"); | 
| + | 
| +    // "The previous region for this ID is an ancestor region of the parent region." | 
| +    if (previousHitRegion && parentHitRegion && previousHitRegion->isAncestorOf(parentHitRegion.get())) | 
| +        exceptionState.throwDOMException(NotSupportedError, "the previous region is an ancestor of the paretn region"); | 
| + | 
| +    // 12. "If the parent member is not null but parent region is null, then | 
| +    // throw a NotFoundError exception and abort these steps." | 
| +    if (!parentId.isNull() && !parentHitRegion) | 
| +        exceptionState.throwDOMException(NotFoundError, "the specified parent region does not exist"); | 
| + | 
| +    // 13. "If any of the following conditions are met, throw a SyntaxError exception and abort these steps." | 
| + | 
| +    // "The arguments object's cursor member is not null but is neither an | 
| +    // ASCII case-insensitive match for the string "inherit", nor a valid CSS | 
| +    // 'cursor' property value." | 
| +    if (!cursor.isNull() && !isValidCursorValue(cursor)) | 
| +        exceptionState.throwDOMException(SyntaxError, "the cursor value (' + cursor + ') is not recognized"); | 
| + | 
| +    // "The arguments object's role member is not null but its value is not an | 
| +    // ordered set of unique space-separated tokens whose tokens are all | 
| +    // case-sensitive matches for names of non-abstract WAI-ARIA roles." | 
| +    if (!role.isNull() && !hasValidAriaRoles(role)) | 
| +        exceptionState.throwDOMException(SyntaxError, "the specified role(s) not valid ARIA role(s)"); | 
| + | 
| +    return !exceptionState.hadException(); | 
| +} | 
| + | 
| +CanvasHitRegion::CanvasHitRegion(const DecodedHitRegionOptions& options) | 
| +    : m_geometryInfo(GeometryInfo::create(options.path, options.fillRule, this)) | 
| +    , m_parent(options.parentHitRegion) | 
| +    , m_control(options.control) | 
| +    , m_id(options.id) | 
| +    , m_cursor(options.cursor) | 
| +    , m_label(options.label) | 
| +    , m_ariaRole(options.role) | 
| +{ | 
| +} | 
| + | 
| +CanvasHitRegion::~CanvasHitRegion() | 
| +{ | 
| +    // Detach ourselves from our GeometryInfo, rendering it unable to produce a | 
| +    // matching result on a hit test (but still occupying the area.) | 
| +    m_geometryInfo->detachHitRegion(); | 
| +} | 
| + | 
| +CanvasHitRegion::GeometryInfo::GeometryInfo(const Path& path, WindRule fillRule, CanvasHitRegion* hitRegion) | 
| +    : m_path(path) | 
| +    , m_fillRule(fillRule) | 
| +    , m_hitRegion(hitRegion) | 
| +{ | 
| +} | 
| + | 
| +bool CanvasHitRegion::GeometryInfo::contains(const LayoutPoint& point) const | 
| +{ | 
| +    if (!m_path.boundingRect().contains(point)) | 
| +        return false; | 
| +    return m_path.contains(point, m_fillRule); | 
| +} | 
| + | 
| +bool CanvasHitRegion::GeometryInfo::isEnclosed(const FloatRect& queryRect, const AffineTransform& queryTransform) const | 
| +{ | 
| +    // Pessimistic 'enclosure' test. | 
| +    FloatRect testRect; | 
| +    if (queryTransform.isIdentity()) { | 
| +        testRect = m_path.boundingRect(); | 
| +    } else { | 
| +        Path inverseTransformedPath = m_path; | 
| +        inverseTransformedPath.transform(queryTransform); | 
| +        testRect = inverseTransformedPath.boundingRect(); | 
| +    } | 
| +    return queryRect.contains(testRect); | 
| +} | 
| + | 
| +bool CanvasHitRegion::isAncestorOf(const CanvasHitRegion* hitRegion) const | 
| +{ | 
| +    if (!hitRegion || hitRegion == this) | 
| +        return false; | 
| + | 
| +    while ((hitRegion = hitRegion->parent())) { | 
| +        if (hitRegion == this) | 
| +            return true; | 
| +    } | 
| +    return false; | 
| +} | 
| + | 
| +void CanvasHitRegion::trace(Visitor* visitor) | 
| +{ | 
| +#if ENABLE(OILPAN) | 
| +    visitor->trace(m_parent); | 
| +    visitor->trace(m_control); | 
| +#endif | 
| +} | 
| + | 
| +void CanvasHitRegionManager::add(PassRefPtrWillBeRawPtr<CanvasHitRegion> passHitRegion) | 
| +{ | 
| +    RefPtrWillBeRawPtr<CanvasHitRegion> hitRegion = passHitRegion; | 
| +    ASSERT(!m_hitRegions.contains(hitRegion)); | 
| +    m_hitRegions.add(hitRegion); | 
| +    // Add to the id -> hit region map. | 
| +    if (!hitRegion->id().isEmpty()) { | 
| +        ASSERT(!m_hitRegionIdMap.contains(hitRegion->id())); | 
| +        m_hitRegionIdMap.set(hitRegion->id(), hitRegion); | 
| +    } | 
| +    // Add the hit region to the spatial query structure. | 
| +    m_hitRegionGeometry.append(hitRegion->geometryInfo()); | 
| +} | 
| + | 
| +void CanvasHitRegionManager::remove(CanvasHitRegion* hitRegion) | 
| +{ | 
| +    if (!hitRegion) | 
| +        return; | 
| + | 
| +    if (!hitRegion->id().isEmpty()) { | 
| +        ASSERT(m_hitRegionIdMap.get(hitRegion->id()) == hitRegion); | 
| +        m_hitRegionIdMap.remove(hitRegion->id()); | 
| +    } | 
| +    ASSERT(m_hitRegions.contains(hitRegion)); | 
| +    m_hitRegions.remove(hitRegion); | 
| +} | 
| + | 
| +void CanvasHitRegionManager::clear() | 
| +{ | 
| +    m_hitRegionGeometry.clear(); | 
| +    m_hitRegionIdMap.clear(); | 
| +    m_hitRegions.clear(); | 
| +} | 
| + | 
| +void CanvasHitRegionManager::removeEnclosed(const FloatRect& rect, const AffineTransform& currentTransform) | 
| +{ | 
| +    size_t geometryCount = m_hitRegionGeometry.size(); | 
| +    if (!geometryCount) | 
| +        return; | 
| + | 
| +    FloatRect queryRect; | 
| +    AffineTransform queryTransform; | 
| +    if (currentTransform.preservesAxisAlignment()) { | 
| +        queryRect = currentTransform.mapRect(rect); | 
| +        // |queryTransform| is identity. | 
| +    } else { | 
| +        queryRect = rect; | 
| +        queryTransform = currentTransform.inverse(); | 
| +    } | 
| + | 
| +    for (size_t i = geometryCount - 1; i >= 0; --i) { | 
| +        CanvasHitRegion::GeometryInfo* geometryInfo = m_hitRegionGeometry[i].get(); | 
| +        if (!geometryInfo->isEnclosed(queryRect, queryTransform)) | 
| +            continue; | 
| + | 
| +        // Drop the associated hit region (if any.) | 
| +        remove(geometryInfo->hitRegion()); | 
| +        // Drop the geometry info. | 
| +        m_hitRegionGeometry.remove(i); | 
| +    } | 
| + | 
| +    // If there's still hit regions referenced from the geometry list, then add | 
| +    // a query rectangle that will consume any queries to that area. | 
| +    if (!m_hitRegions.isEmpty()) { | 
| +        Path exclusionPath; | 
| +        exclusionPath.addRect(rect); | 
| +        exclusionPath.transform(currentTransform); | 
| +        m_hitRegionGeometry.append(CanvasHitRegion::GeometryInfo::create(exclusionPath, RULE_NONZERO, 0)); | 
| +    } else { | 
| +        // Clear the geometry list, since it will only contain exclusions. | 
| +        m_hitRegionGeometry.clear(); | 
| +    } | 
| +} | 
| + | 
| +CanvasHitRegion* CanvasHitRegionManager::getHitRegionById(const String& id) const | 
| +{ | 
| +    return m_hitRegionIdMap.get(id); | 
| +} | 
| + | 
| +CanvasHitRegion* CanvasHitRegionManager::getHitRegionAtPoint(const LayoutPoint& point) const | 
| +{ | 
| +    HitRegionGeometry::const_reverse_iterator itEnd = m_hitRegionGeometry.rend(); | 
| +    for (HitRegionGeometry::const_reverse_iterator it = m_hitRegionGeometry.rbegin(); it != itEnd; ++it) { | 
| +        if ((*it)->contains(point)) | 
| +            return (*it)->hitRegion(); | 
| +    } | 
| +    return 0; | 
| +} | 
| + | 
| +void CanvasHitRegionManager::trace(Visitor* visitor) | 
| +{ | 
| +#if ENABLE(OILPAN) | 
| +    visitor->trace(m_hitRegions); | 
| +    visitor->trace(m_hitRegionIdMap); | 
| +#endif | 
| +} | 
| + | 
| +} // namespace WebCore | 
|  |