KWQAccObjectCache.mm   [plain text]


/*
 * Copyright (C) 2003 Apple Computer, 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 COMPUTER, 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 COMPUTER, 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 "KWQDOMNode.h"
#include "KWQAccObjectCache.h"
#include "KWQAccObject.h"
#include "KWQAssertions.h"
#include "KWQFoundationExtras.h"
#include <qstring.h>
#include <render_object.h>

using khtml::EAffinity;
using khtml::RenderObject;
using khtml::VisiblePosition;

// The simple Cocoa calls in this file can't throw.

bool KWQAccObjectCache::gAccessibilityEnabled = false;

typedef struct KWQTextMarkerData  {
    KWQAccObjectID  accObjectID;
    DOM::NodeImpl*  nodeImpl;
    int             offset;
    EAffinity       affinity;
};

KWQAccObjectCache::KWQAccObjectCache()
{
    accCache = NULL;
    accCacheByID = NULL;
    accObjectIDSource = 0;
}

KWQAccObjectCache::~KWQAccObjectCache()
{
    // Destroy the dictionary
    if (accCache)
        CFRelease(accCache);
        
    // Destroy the ID lookup dictionary
    if (accCacheByID) {
        // accCacheByID should have been emptied by releasing accCache
        ASSERT(CFDictionaryGetCount(accCacheByID) == 0);
        CFRelease(accCacheByID);
    }
}

KWQAccObject* KWQAccObjectCache::accObject(RenderObject* renderer)
{
    if (!accCache)
        // No need to retain/free either impl key, or id value.
        accCache = CFDictionaryCreateMutable(NULL, 0, NULL, NULL);
    
    KWQAccObject* obj = (KWQAccObject*)CFDictionaryGetValue(accCache, renderer);
    if (!obj) {
        obj = [[KWQAccObject alloc] initWithRenderer: renderer]; // Initial ref happens here.
        setAccObject(renderer, obj);
    }

    return obj;
}

void KWQAccObjectCache::setAccObject(RenderObject* impl, KWQAccObject* accObject)
{
    if (!accCache)
        // No need to retain/free either impl key, or id value.
        accCache = CFDictionaryCreateMutable(NULL, 0, NULL, NULL);
    
    CFDictionarySetValue(accCache, (const void *)impl, accObject);
    ASSERT(!accCacheByID || CFDictionaryGetCount(accCache) >= CFDictionaryGetCount(accCacheByID));
}

void KWQAccObjectCache::removeAccObject(RenderObject* impl)
{
    if (!accCache)
        return;

    KWQAccObject* obj = (KWQAccObject*)CFDictionaryGetValue(accCache, impl);
    if (obj) {
        [obj detach];
        [obj release];
        CFDictionaryRemoveValue(accCache, impl);
    }

    ASSERT(!accCacheByID || CFDictionaryGetCount(accCache) >= CFDictionaryGetCount(accCacheByID));
}

KWQAccObjectID KWQAccObjectCache::getAccObjectID(KWQAccObject* accObject)
{
    KWQAccObjectID  accObjectID;

    // create the ID table as needed
    if (accCacheByID == NULL)
        accCacheByID = CFDictionaryCreateMutable(NULL, 0, NULL, NULL);
    
    // check for already-assigned ID
    accObjectID = [accObject accObjectID];
    if (accObjectID != 0) {
        ASSERT(CFDictionaryContainsKey(accCacheByID, (const void *)accObjectID));
        return accObjectID;
    }
    
    // generate a new ID
    accObjectID = accObjectIDSource + 1;
    while (accObjectID == 0 || CFDictionaryContainsKey(accCacheByID, (const void *)accObjectID)) {
        ASSERT(accObjectID != accObjectIDSource);   // check for exhaustion
        accObjectID += 1;
    }
    accObjectIDSource = accObjectID;

    // assign the new ID to the object
    [accObject setAccObjectID:accObjectID];
    
    // add the object to the ID table
    CFDictionarySetValue(accCacheByID, (const void *)accObjectID, accObject);
    ASSERT(CFDictionaryContainsKey(accCacheByID, (const void *)accObjectID));
    
    return accObjectID;
}

void KWQAccObjectCache::removeAccObjectID(KWQAccObject* accObject)
{
    // retrieve and clear the ID from the object, nothing to do if it was never assigned
    KWQAccObjectID  accObjectID = [accObject accObjectID];
    if (accObjectID == 0)
        return;
    [accObject setAccObjectID:0];
    
    // remove the element from the lookup table
    ASSERT(accCacheByID != NULL);
    ASSERT(CFDictionaryContainsKey(accCacheByID, (const void *)accObjectID));
    CFDictionaryRemoveValue(accCacheByID, (const void *)accObjectID);
}

#if OMIT_TIGER_FEATURES
// no parameterized attributes in Panther... they were introduced in Tiger
#else
AXTextMarkerRef   KWQAccObjectCache::textMarkerForVisiblePosition (const VisiblePosition & visiblePos)
{
    KWQTextMarkerData   textMarkerData;
    AXTextMarkerRef     textMarker = NULL;    

    DOM::Position deepPos = visiblePos.deepEquivalent();
    DOM::NodeImpl* domNode = deepPos.node();
    if (domNode == NULL) {
        ASSERT(domNode != NULL);
        return NULL;
    }
    
    // locate the renderer, which must exist for a visible dom node
    khtml::RenderObject* renderer = domNode->renderer();
    ASSERT(renderer != NULL);
    
    // find or create an accessibility object for this renderer
    KWQAccObject* accObject = this->accObject(renderer);
    
    // create a text marker, adding an ID for the KWQAccObject if needed
    textMarkerData.accObjectID = getAccObjectID(accObject);
    textMarkerData.nodeImpl = domNode;
    textMarkerData.offset = deepPos.offset();
    textMarkerData.affinity = visiblePos.affinity();
    textMarker = AXTextMarkerCreate(NULL, (const UInt8*)&textMarkerData, sizeof(textMarkerData));

    // autorelease it because we will never see it again
    KWQCFAutorelease(textMarker);
    return textMarker; 
}

VisiblePosition   KWQAccObjectCache::visiblePositionForTextMarker (AXTextMarkerRef textMarker)
{
    KWQTextMarkerData*  textMarkerData;
    
    // catch some bad inputs
    if (textMarker == NULL)
        return VisiblePosition();
    
    if (AXTextMarkerGetLength(textMarker) != sizeof(KWQTextMarkerData)) {
        ASSERT (AXTextMarkerGetLength(textMarker) == sizeof(KWQTextMarkerData));
        return VisiblePosition();
    }
    
    textMarkerData = (KWQTextMarkerData*) AXTextMarkerGetBytePtr(textMarker);
    if (textMarkerData == NULL) {
        ASSERT(textMarkerData != NULL);
        return VisiblePosition();
    }

    // return empty position if the text marker is no longer valid
    if (!accCacheByID || !CFDictionaryContainsKey(accCacheByID, (const void *)textMarkerData->accObjectID))
        return VisiblePosition();

    // return the position from the data we stored earlier
    return VisiblePosition(textMarkerData->nodeImpl, textMarkerData->offset, textMarkerData->affinity);
}
#endif

void KWQAccObjectCache::detach(RenderObject* renderer)
{
    removeAccObject(renderer);
}

void KWQAccObjectCache::childrenChanged(RenderObject* renderer)
{
    if (!accCache)
        return;
    
    KWQAccObject* obj = (KWQAccObject*)CFDictionaryGetValue(accCache, renderer);
    if (!obj)
        return;
    
    [obj childrenChanged];
}

void KWQAccObjectCache::postNotificationToTopWebArea(RenderObject* renderer, const QString& msg)
{
    if (renderer) {
        RenderObject * obj = renderer->document()->topDocument()->renderer();
        NSAccessibilityPostNotification(accObject(obj), msg.getNSString());
    }
}

void KWQAccObjectCache::postNotification(RenderObject* renderer, const QString& msg)
{
    if (renderer)
        NSAccessibilityPostNotification(accObject(renderer), msg.getNSString());
}

extern "C" void NSAccessibilityHandleFocusChanged(void);
void KWQAccObjectCache::handleFocusedUIElementChanged(void)
{
    // This is an internal AppKit call that does a number of things in addition to
    // sending the AXFocusedUIElementChanged notification.  It will call
    // will call accessibilityFocusedUIElement() to determine which element
    // to include in the notification.
    NSAccessibilityHandleFocusChanged();
}