mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
04083ef9b4
- Bug 1235261 - Part 1: Rename nsAutoTArray to AutoTArray. r=froydnj (0662c2ac56)
- Bug 1235261 - Part 2: Switch some uses of AutoFallibleTArray to AutoTArray. r=froydnj (ab52085f2a)
- Bug 1235261 - Part 3: Switch remaining uses of AutoFallibleTArray to AutoTArray. r=froydnj (3763b16ddd)
- Bug 1235261 - Part 4: Remove AutoFallibleTArray. r=froydnj (5480b0d786)
- Bug 1235261 - Part 5: Merge nsAutoArrayBase into AutoTArray. r=froydnj (6c64e73e3b)
- Bug 1235261 - Part 7: Remove AutoInfallibleTArray. r=froydnj (acf266464e)
- Bug 1222624: Make xpath document() function use nsIPrincipals and nsIURIs rather than strings. r=peterv (5ee694d132)
- Bug 1235261 - Part 6: Rename AutoInfallibleTArray to AutoTArray. r=froydnj (d282f7df6c)
- Bug 1241394 - Hit testing with 3d transforms should use fuzzy when comparing depths. r=thinker (6c3f50670f)
- Bug 1241394 - Follow up to fix windows build bustage. (02ab2600af)
- Bug 1241394 - Check clip for the children of the establisher. r=mattwoodrow (46f151ea55)
- bug 1241453 - allow caching proxies in xpcAccessibleDocuments r=davidb (f5d41ad2ee)
- Bug 1247364 - use AllChildrenIterator::Seek by a11y tree walker, r=davidb (0ec230908e)
- Bug 1248840 - rename TreeWalker::NextChild, r=yzen (c89ecc5a29)
- Bug 1249927 - devirtualize CanHavaAnonymousChildren, r=davdib (89e8088e63)
- Bug 1206598 - Use universal reference to reduce the redundant copy. r=nfroyd (bae4ad6dd1)
- Bug 1247364 - add AllChildrenIterator::Seek, r=bz (215abebf12)
- bug 1241453 - allow storing proxies in xpcAccessibleGeneric::mIntl r=davidb (dd5e6c896b)
- bug 1241453 - allow constructing xpcAccessibles with proxies r=davidb (d0258122be)
- bug 1241453 - fixup xpcAccessible Intl() methods to not assume mIntl is always an Accessible r=davidb (168f71fdf5)
- bug 1241453 - allow xpcAccessibleDocument::mCache to use proxies as keys r=davidb (85b7eec81c)
- bug 1241453 - assert accessibles are only added to non remote xpcAccessibleDocuments r=davidb (7731b21d17)
- bug 1243077 - add ToXPC{,Document} overloads for proxied accessibles r=davidb (7bc085f1b5)
- bug 1243077 - add AccessibleOrProxy xpcAccessible::IntlGeneric() r=davidb (006a635992)
- Bug 1245464 - initialize with 0 mSupportedIfaces in xpcAccessibleGeneric in order to avoid corrupted result after bit-wise operation. r=surkov (ae41bafcef)
- bug 1241453 - allow caching xpc documents for remote documents r=davidb (a357630690)
- bug 1241453 - factor dispatching nsIAccessibleEvents out of HandleAccEvent() r=davidb (091073d981)
- Bug 1249183 - Suppress GC harder, r=terrence (2185ccb4dd)
- Bug 1248420 - Handle JSObject::getGroup OOM in js::ArraySetLength. r=jandem (04b67c8d31)
- Bug 1242270 - Add SPS pseudo frames for the Array.prototype methods; r=shu (f5e5871439)
- Bug 1247701 - Bail from ArrayShiftDenseKernel if the array is used by for-in iteration. r=jandem (41eff38954)
- Bug 1247701 followup - Change ArrayShiftDenseKernel to receive handle. r=bz (b29ce0c555)
630 lines
19 KiB
C++
630 lines
19 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "InterfaceInitFuncs.h"
|
|
#include "mozilla/a11y/PDocAccessible.h"
|
|
#include "Accessible-inl.h"
|
|
#include "HyperTextAccessible-inl.h"
|
|
#include "nsMai.h"
|
|
#include "ProxyAccessible.h"
|
|
|
|
#include "nsIAccessibleTypes.h"
|
|
#include "nsIPersistentProperties2.h"
|
|
#include "nsISimpleEnumerator.h"
|
|
|
|
#include "mozilla/Likely.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::a11y;
|
|
|
|
static const char* sAtkTextAttrNames[ATK_TEXT_ATTR_LAST_DEFINED];
|
|
|
|
void
|
|
ConvertTextAttributeToAtkAttribute(const nsACString& aName,
|
|
const nsAString& aValue,
|
|
AtkAttributeSet** aAttributeSet)
|
|
{
|
|
// Handle attributes where atk has its own name.
|
|
const char* atkName = nullptr;
|
|
nsAutoString atkValue;
|
|
if (aName.EqualsLiteral("color")) {
|
|
// The format of the atk attribute is r,g,b and the gecko one is
|
|
// rgb(r, g, b).
|
|
atkValue = Substring(aValue, 4, aValue.Length() - 5);
|
|
atkValue.StripWhitespace();
|
|
atkName = sAtkTextAttrNames[ATK_TEXT_ATTR_FG_COLOR];
|
|
} else if (aName.EqualsLiteral("background-color")) {
|
|
// The format of the atk attribute is r,g,b and the gecko one is
|
|
// rgb(r, g, b).
|
|
atkValue = Substring(aValue, 4, aValue.Length() - 5);
|
|
atkValue.StripWhitespace();
|
|
atkName = sAtkTextAttrNames[ATK_TEXT_ATTR_BG_COLOR];
|
|
} else if (aName.EqualsLiteral("font-family")) {
|
|
atkValue = aValue;
|
|
atkName = sAtkTextAttrNames[ATK_TEXT_ATTR_FAMILY_NAME];
|
|
} else if (aName.EqualsLiteral("font-size")) {
|
|
// ATK wants the number of pixels without px at the end.
|
|
atkValue = StringHead(aValue, aValue.Length() - 2);
|
|
atkName = sAtkTextAttrNames[ATK_TEXT_ATTR_SIZE];
|
|
} else if (aName.EqualsLiteral("font-weight")) {
|
|
atkValue = aValue;
|
|
atkName = sAtkTextAttrNames[ATK_TEXT_ATTR_WEIGHT];
|
|
} else if (aName.EqualsLiteral("invalid")) {
|
|
atkValue = aValue;
|
|
atkName = sAtkTextAttrNames[ATK_TEXT_ATTR_INVALID];
|
|
}
|
|
|
|
if (atkName) {
|
|
AtkAttribute* objAttr =
|
|
static_cast<AtkAttribute*>(g_malloc(sizeof(AtkAttribute)));
|
|
objAttr->name = g_strdup(atkName);
|
|
objAttr->value = g_strdup(NS_ConvertUTF16toUTF8(atkValue).get());
|
|
*aAttributeSet = g_slist_prepend(*aAttributeSet, objAttr);
|
|
}
|
|
}
|
|
|
|
static AtkAttributeSet*
|
|
ConvertToAtkTextAttributeSet(nsTArray<Attribute>& aAttributes)
|
|
{
|
|
AtkAttributeSet* objAttributeSet = nullptr;
|
|
for (size_t i = 0; i < aAttributes.Length(); ++i) {
|
|
AtkAttribute* objAttr = (AtkAttribute *)g_malloc(sizeof(AtkAttribute));
|
|
objAttr->name = g_strdup(aAttributes[i].Name().get());
|
|
objAttr->value =
|
|
g_strdup(NS_ConvertUTF16toUTF8(aAttributes[i].Value()).get());
|
|
objAttributeSet = g_slist_prepend(objAttributeSet, objAttr);
|
|
ConvertTextAttributeToAtkAttribute(aAttributes[i].Name(),
|
|
aAttributes[i].Value(),
|
|
&objAttributeSet);
|
|
}
|
|
return objAttributeSet;
|
|
}
|
|
|
|
static AtkAttributeSet*
|
|
ConvertToAtkTextAttributeSet(nsIPersistentProperties* aAttributes)
|
|
{
|
|
if (!aAttributes)
|
|
return nullptr;
|
|
|
|
AtkAttributeSet* objAttributeSet = nullptr;
|
|
nsCOMPtr<nsISimpleEnumerator> propEnum;
|
|
nsresult rv = aAttributes->Enumerate(getter_AddRefs(propEnum));
|
|
NS_ENSURE_SUCCESS(rv, nullptr);
|
|
|
|
bool hasMore = false;
|
|
while (NS_SUCCEEDED(propEnum->HasMoreElements(&hasMore)) && hasMore) {
|
|
nsCOMPtr<nsISupports> sup;
|
|
rv = propEnum->GetNext(getter_AddRefs(sup));
|
|
NS_ENSURE_SUCCESS(rv, objAttributeSet);
|
|
|
|
nsCOMPtr<nsIPropertyElement> propElem(do_QueryInterface(sup));
|
|
NS_ENSURE_TRUE(propElem, objAttributeSet);
|
|
|
|
nsAutoCString name;
|
|
rv = propElem->GetKey(name);
|
|
NS_ENSURE_SUCCESS(rv, objAttributeSet);
|
|
|
|
nsAutoString value;
|
|
rv = propElem->GetValue(value);
|
|
NS_ENSURE_SUCCESS(rv, objAttributeSet);
|
|
|
|
AtkAttribute* objAttr = (AtkAttribute*)g_malloc(sizeof(AtkAttribute));
|
|
objAttr->name = g_strdup(name.get());
|
|
objAttr->value = g_strdup(NS_ConvertUTF16toUTF8(value).get());
|
|
objAttributeSet = g_slist_prepend(objAttributeSet, objAttr);
|
|
|
|
ConvertTextAttributeToAtkAttribute(name, value, &objAttributeSet);
|
|
}
|
|
|
|
// libatk-adaptor will free it
|
|
return objAttributeSet;
|
|
}
|
|
|
|
static void
|
|
ConvertTexttoAsterisks(AccessibleWrap* accWrap, nsAString& aString)
|
|
{
|
|
// convert each char to "*" when it's "password text"
|
|
if (accWrap->NativeRole() == roles::PASSWORD_TEXT) {
|
|
for (uint32_t i = 0; i < aString.Length(); i++)
|
|
aString.Replace(i, 1, NS_LITERAL_STRING("*"));
|
|
}
|
|
}
|
|
|
|
extern "C" {
|
|
|
|
static gchar*
|
|
getTextCB(AtkText *aText, gint aStartOffset, gint aEndOffset)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
nsAutoString autoStr;
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole())
|
|
return nullptr;
|
|
|
|
text->TextSubstring(aStartOffset, aEndOffset, autoStr);
|
|
|
|
ConvertTexttoAsterisks(accWrap, autoStr);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
proxy->TextSubstring(aStartOffset, aEndOffset, autoStr);
|
|
}
|
|
|
|
NS_ConvertUTF16toUTF8 cautoStr(autoStr);
|
|
|
|
//copy and return, libspi will free it.
|
|
return (cautoStr.get()) ? g_strdup(cautoStr.get()) : nullptr;
|
|
}
|
|
|
|
static gchar*
|
|
getTextAfterOffsetCB(AtkText *aText, gint aOffset,
|
|
AtkTextBoundary aBoundaryType,
|
|
gint *aStartOffset, gint *aEndOffset)
|
|
{
|
|
nsAutoString autoStr;
|
|
int32_t startOffset = 0, endOffset = 0;
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole())
|
|
return nullptr;
|
|
|
|
text->TextAfterOffset(aOffset, aBoundaryType, &startOffset, &endOffset, autoStr);
|
|
ConvertTexttoAsterisks(accWrap, autoStr);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
proxy->GetTextAfterOffset(aOffset, aBoundaryType, autoStr, &startOffset,
|
|
&endOffset);
|
|
}
|
|
|
|
*aStartOffset = startOffset;
|
|
*aEndOffset = endOffset;
|
|
|
|
NS_ConvertUTF16toUTF8 cautoStr(autoStr);
|
|
return (cautoStr.get()) ? g_strdup(cautoStr.get()) : nullptr;
|
|
}
|
|
|
|
static gchar*
|
|
getTextAtOffsetCB(AtkText *aText, gint aOffset,
|
|
AtkTextBoundary aBoundaryType,
|
|
gint *aStartOffset, gint *aEndOffset)
|
|
{
|
|
nsAutoString autoStr;
|
|
int32_t startOffset = 0, endOffset = 0;
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole())
|
|
return nullptr;
|
|
|
|
text->TextAtOffset(aOffset, aBoundaryType, &startOffset, &endOffset, autoStr);
|
|
ConvertTexttoAsterisks(accWrap, autoStr);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
proxy->GetTextAtOffset(aOffset, aBoundaryType, autoStr, &startOffset,
|
|
&endOffset);
|
|
}
|
|
|
|
*aStartOffset = startOffset;
|
|
*aEndOffset = endOffset;
|
|
|
|
NS_ConvertUTF16toUTF8 cautoStr(autoStr);
|
|
return (cautoStr.get()) ? g_strdup(cautoStr.get()) : nullptr;
|
|
}
|
|
|
|
static gunichar
|
|
getCharacterAtOffsetCB(AtkText* aText, gint aOffset)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return 0;
|
|
}
|
|
|
|
// char16_t is unsigned short in Mozilla, gnuichar is guint32 in glib.
|
|
return static_cast<gunichar>(text->CharAt(aOffset));
|
|
}
|
|
|
|
if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
return static_cast<gunichar>(proxy->CharAt(aOffset));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static gchar*
|
|
getTextBeforeOffsetCB(AtkText *aText, gint aOffset,
|
|
AtkTextBoundary aBoundaryType,
|
|
gint *aStartOffset, gint *aEndOffset)
|
|
{
|
|
nsAutoString autoStr;
|
|
int32_t startOffset = 0, endOffset = 0;
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole())
|
|
return nullptr;
|
|
|
|
text->TextBeforeOffset(aOffset, aBoundaryType,
|
|
&startOffset, &endOffset, autoStr);
|
|
ConvertTexttoAsterisks(accWrap, autoStr);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
proxy->GetTextBeforeOffset(aOffset, aBoundaryType, autoStr, &startOffset,
|
|
&endOffset);
|
|
}
|
|
|
|
*aStartOffset = startOffset;
|
|
*aEndOffset = endOffset;
|
|
|
|
NS_ConvertUTF16toUTF8 cautoStr(autoStr);
|
|
return (cautoStr.get()) ? g_strdup(cautoStr.get()) : nullptr;
|
|
}
|
|
|
|
static gint
|
|
getCaretOffsetCB(AtkText *aText)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return 0;
|
|
}
|
|
|
|
return static_cast<gint>(text->CaretOffset());
|
|
}
|
|
|
|
if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
return static_cast<gint>(proxy->CaretOffset());
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static AtkAttributeSet*
|
|
getRunAttributesCB(AtkText *aText, gint aOffset,
|
|
gint *aStartOffset,
|
|
gint *aEndOffset)
|
|
{
|
|
*aStartOffset = -1;
|
|
*aEndOffset = -1;
|
|
int32_t startOffset = 0, endOffset = 0;
|
|
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsCOMPtr<nsIPersistentProperties> attributes =
|
|
text->TextAttributes(false, aOffset, &startOffset, &endOffset);
|
|
|
|
*aStartOffset = startOffset;
|
|
*aEndOffset = endOffset;
|
|
|
|
return ConvertToAtkTextAttributeSet(attributes);
|
|
}
|
|
|
|
ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText));
|
|
if (!proxy) {
|
|
return nullptr;
|
|
}
|
|
|
|
AutoTArray<Attribute, 10> attrs;
|
|
proxy->TextAttributes(false, aOffset, &attrs, &startOffset, &endOffset);
|
|
*aStartOffset = startOffset;
|
|
*aEndOffset = endOffset;
|
|
return ConvertToAtkTextAttributeSet(attrs);
|
|
}
|
|
|
|
static AtkAttributeSet*
|
|
getDefaultAttributesCB(AtkText *aText)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsCOMPtr<nsIPersistentProperties> attributes = text->DefaultTextAttributes();
|
|
return ConvertToAtkTextAttributeSet(attributes);
|
|
}
|
|
|
|
ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText));
|
|
if (!proxy) {
|
|
return nullptr;
|
|
}
|
|
|
|
AutoTArray<Attribute, 10> attrs;
|
|
proxy->DefaultTextAttributes(&attrs);
|
|
return ConvertToAtkTextAttributeSet(attrs);
|
|
}
|
|
|
|
static void
|
|
getCharacterExtentsCB(AtkText *aText, gint aOffset,
|
|
gint *aX, gint *aY,
|
|
gint *aWidth, gint *aHeight,
|
|
AtkCoordType aCoords)
|
|
{
|
|
if(!aX || !aY || !aWidth || !aHeight) {
|
|
return;
|
|
}
|
|
|
|
nsIntRect rect;
|
|
uint32_t geckoCoordType;
|
|
if (aCoords == ATK_XY_SCREEN) {
|
|
geckoCoordType = nsIAccessibleCoordinateType::COORDTYPE_SCREEN_RELATIVE;
|
|
} else {
|
|
geckoCoordType = nsIAccessibleCoordinateType::COORDTYPE_WINDOW_RELATIVE;
|
|
}
|
|
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return;
|
|
}
|
|
|
|
rect = text->CharBounds(aOffset, geckoCoordType);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
rect = proxy->CharBounds(aOffset, geckoCoordType);
|
|
} else {
|
|
return;
|
|
}
|
|
|
|
*aX = rect.x;
|
|
*aY = rect.y;
|
|
*aWidth = rect.width;
|
|
*aHeight = rect.height;
|
|
}
|
|
|
|
static void
|
|
getRangeExtentsCB(AtkText *aText, gint aStartOffset, gint aEndOffset,
|
|
AtkCoordType aCoords, AtkTextRectangle *aRect)
|
|
{
|
|
if (!aRect) {
|
|
return;
|
|
}
|
|
|
|
nsIntRect rect;
|
|
uint32_t geckoCoordType;
|
|
if (aCoords == ATK_XY_SCREEN) {
|
|
geckoCoordType = nsIAccessibleCoordinateType::COORDTYPE_SCREEN_RELATIVE;
|
|
} else {
|
|
geckoCoordType = nsIAccessibleCoordinateType::COORDTYPE_WINDOW_RELATIVE;
|
|
}
|
|
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if(accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return;
|
|
}
|
|
|
|
rect = text->TextBounds(aStartOffset, aEndOffset, geckoCoordType);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
rect = proxy->TextBounds(aStartOffset, aEndOffset, geckoCoordType);
|
|
} else {
|
|
return;
|
|
}
|
|
|
|
aRect->x = rect.x;
|
|
aRect->y = rect.y;
|
|
aRect->width = rect.width;
|
|
aRect->height = rect.height;
|
|
}
|
|
|
|
static gint
|
|
getCharacterCountCB(AtkText *aText)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* textAcc = accWrap->AsHyperText();
|
|
return
|
|
textAcc->IsDefunct() ? 0 : static_cast<gint>(textAcc->CharacterCount());
|
|
}
|
|
|
|
if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
return proxy->CharacterCount();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static gint
|
|
getOffsetAtPointCB(AtkText *aText,
|
|
gint aX, gint aY,
|
|
AtkCoordType aCoords)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return -1;
|
|
}
|
|
|
|
return static_cast<gint>(
|
|
text->OffsetAtPoint(aX, aY,
|
|
(aCoords == ATK_XY_SCREEN ?
|
|
nsIAccessibleCoordinateType::COORDTYPE_SCREEN_RELATIVE :
|
|
nsIAccessibleCoordinateType::COORDTYPE_WINDOW_RELATIVE)));
|
|
}
|
|
|
|
if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
return static_cast<gint>(
|
|
proxy->OffsetAtPoint(aX, aY,
|
|
(aCoords == ATK_XY_SCREEN ?
|
|
nsIAccessibleCoordinateType::COORDTYPE_SCREEN_RELATIVE :
|
|
nsIAccessibleCoordinateType::COORDTYPE_WINDOW_RELATIVE)));
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
static gint
|
|
getTextSelectionCountCB(AtkText *aText)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return 0;
|
|
}
|
|
|
|
return text->SelectionCount();
|
|
}
|
|
|
|
if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
return proxy->SelectionCount();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static gchar*
|
|
getTextSelectionCB(AtkText *aText, gint aSelectionNum,
|
|
gint *aStartOffset, gint *aEndOffset)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
int32_t startOffset = 0, endOffset = 0;
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return nullptr;
|
|
}
|
|
|
|
text->SelectionBoundsAt(aSelectionNum, &startOffset, &endOffset);
|
|
*aStartOffset = startOffset;
|
|
*aEndOffset = endOffset;
|
|
|
|
return getTextCB(aText, *aStartOffset, *aEndOffset);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
nsString data;
|
|
proxy->SelectionBoundsAt(aSelectionNum, data, &startOffset, &endOffset);
|
|
*aStartOffset = startOffset;
|
|
*aEndOffset = endOffset;
|
|
|
|
NS_ConvertUTF16toUTF8 dataAsUTF8(data);
|
|
return (dataAsUTF8.get()) ? g_strdup(dataAsUTF8.get()) : nullptr;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
// set methods
|
|
static gboolean
|
|
addTextSelectionCB(AtkText *aText,
|
|
gint aStartOffset,
|
|
gint aEndOffset)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return FALSE;
|
|
}
|
|
|
|
return text->AddToSelection(aStartOffset, aEndOffset);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
return proxy->AddToSelection(aStartOffset, aEndOffset);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
removeTextSelectionCB(AtkText *aText,
|
|
gint aSelectionNum)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return FALSE;
|
|
}
|
|
|
|
return text->RemoveFromSelection(aSelectionNum);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
return proxy->RemoveFromSelection(aSelectionNum);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
setTextSelectionCB(AtkText *aText, gint aSelectionNum,
|
|
gint aStartOffset, gint aEndOffset)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole()) {
|
|
return FALSE;
|
|
}
|
|
|
|
return text->SetSelectionBoundsAt(aSelectionNum, aStartOffset, aEndOffset);
|
|
} else if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
return proxy->SetSelectionBoundsAt(aSelectionNum, aStartOffset, aEndOffset);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
setCaretOffsetCB(AtkText *aText, gint aOffset)
|
|
{
|
|
AccessibleWrap* accWrap = GetAccessibleWrap(ATK_OBJECT(aText));
|
|
if (accWrap) {
|
|
HyperTextAccessible* text = accWrap->AsHyperText();
|
|
if (!text || !text->IsTextRole() || !text->IsValidOffset(aOffset)) {
|
|
return FALSE;
|
|
}
|
|
|
|
text->SetCaretOffset(aOffset);
|
|
return TRUE;
|
|
}
|
|
|
|
if (ProxyAccessible* proxy = GetProxy(ATK_OBJECT(aText))) {
|
|
proxy->SetCaretOffset(aOffset);
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
void
|
|
textInterfaceInitCB(AtkTextIface* aIface)
|
|
{
|
|
NS_ASSERTION(aIface, "Invalid aIface");
|
|
if (MOZ_UNLIKELY(!aIface))
|
|
return;
|
|
|
|
aIface->get_text = getTextCB;
|
|
aIface->get_text_after_offset = getTextAfterOffsetCB;
|
|
aIface->get_text_at_offset = getTextAtOffsetCB;
|
|
aIface->get_character_at_offset = getCharacterAtOffsetCB;
|
|
aIface->get_text_before_offset = getTextBeforeOffsetCB;
|
|
aIface->get_caret_offset = getCaretOffsetCB;
|
|
aIface->get_run_attributes = getRunAttributesCB;
|
|
aIface->get_default_attributes = getDefaultAttributesCB;
|
|
aIface->get_character_extents = getCharacterExtentsCB;
|
|
aIface->get_range_extents = getRangeExtentsCB;
|
|
aIface->get_character_count = getCharacterCountCB;
|
|
aIface->get_offset_at_point = getOffsetAtPointCB;
|
|
aIface->get_n_selections = getTextSelectionCountCB;
|
|
aIface->get_selection = getTextSelectionCB;
|
|
|
|
// set methods
|
|
aIface->add_selection = addTextSelectionCB;
|
|
aIface->remove_selection = removeTextSelectionCB;
|
|
aIface->set_selection = setTextSelectionCB;
|
|
aIface->set_caret_offset = setCaretOffsetCB;
|
|
|
|
// Cache the string values of the atk text attribute names.
|
|
for (uint32_t i = 0; i < ArrayLength(sAtkTextAttrNames); i++)
|
|
sAtkTextAttrNames[i] =
|
|
atk_text_attribute_get_name(static_cast<AtkTextAttribute>(i));
|
|
}
|