Viewing file: flat_review.py (54.42 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
# Orca # # Copyright 2005-2006 Sun Microsystems Inc. # # 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; if not, write to the # Free Software Foundation, Inc., 59 Temple Place - Suite 330, # Boston, MA 02111-1307, USA.
"""Provides the default implementation for flat review for Orca."""
__id__ = "$Id: flat_review.py,v 1.41 2006/08/10 18:29:56 wwalker Exp $" __version__ = "$Revision: 1.41 $" __date__ = "$Date: 2006/08/10 18:29:56 $" __copyright__ = "Copyright (c) 2005-2006 Sun Microsystems Inc." __license__ = "LGPL"
import re import sys
import atspi import braille import debug import eventsynthesizer import rolenames import util
# [[[WDW - HACK Regular expression to split strings on whitespace # boundaries, which is what we'll use for word dividers instead of # living at the whim of whomever decided to implement the AT-SPI # interfaces for their toolkit or app.]]] # whitespace_re = re.compile(r'(\s+)', re.DOTALL | re.IGNORECASE | re.M)
class Char: """Represents a single char of an Accessibility_Text object."""
def __init__(self, word, index, string, x, y, width, height): """Creates a new char.
Arguments: - word: the Word instance this belongs to - index: the index of this char in the word - string: the actual char - x, y, width, height: the extents of this Char on the screen """
self.word = word self.string = string self.index = index self.x = x self.y = y self.width = width self.height = height
class Word: """Represents a single word of an Accessibility_Text object, or the entire name of an Image or Component if the associated object does not implement the Accessibility_Text interface. As a rule of thumb, all words derived from an Accessibility_Text interface will start with the word and will end with all chars up to the beginning of the next word. That is, whitespace and punctuation will usually be tacked on to the end of words."""
def __init__(self, zone, index, startOffset, string, x, y, width, height): """Creates a new Word.
Arguments: - zone: the Zone instance this belongs to - index: the index of this word in the Zone - string: the actual string - x, y, width, height: the extents of this Char on the screen"""
self.zone = zone self.index = index self.startOffset = startOffset self.string = string self.length = len(string) self.x = x self.y = y self.width = width self.height = height
def __getattr__(self, attr): """Used for lazily determining the chars of a word. We do this to reduce the total number of round trip calls to the app, and to also spread the round trip calls out over the lifetime of a flat review context.
Arguments: - attr: a string indicating the attribute name to retrieve
Returns the value of the given attribute. """
if attr == "chars": if isinstance(self.zone, TextZone): text = self.zone.accessible.text self.chars = [] i = 0 while i < self.length: [char, startOffset, endOffset] = text.getTextAtOffset( self.startOffset + i, atspi.Accessibility.TEXT_BOUNDARY_CHAR) [x, y, width, height] = text.getRangeExtents( startOffset, endOffset, 0) self.chars.append(Char(self, i, char, x, y, width, height)) i += 1 else: self.chars = None return self.chars elif attr.startswith('__') and attr.endswith('__'): raise AttributeError, attr else: return self.__dict__[attr]
class Zone: """Represents text that is a portion of a single horizontal line."""
def __init__(self, accessible, string, x, y, width, height): """Creates a new Zone, which is a horizontal region of text.
Arguments: - accessible: the Accessible associated with this Zone - string: the string being displayed for this Zone - extents: x, y, width, height in screen coordinates """
self.accessible = accessible self.string = string self.length = len(string) self.x = x self.y = y self.width = width self.height = height
def __getattr__(self, attr): """Used for lazily determining the words in a Zone.
Arguments: - attr: a string indicating the attribute name to retrieve
Returns the value of the given attribute. """
if attr == "words": self.words = [] return self.words elif attr.startswith('__') and attr.endswith('__'): raise AttributeError, attr else: return self.__dict__[attr]
def onSameLine(self, zone): """Returns True if this Zone is on the same horiztonal line as the given zone."""
highestBottom = min(self.y + self.height, zone.y + zone.height) lowestTop = max(self.y, zone.y)
# If we do overlap, lets see how much. We'll require a 25% overlap # for now... # if lowestTop < highestBottom: overlapAmount = highestBottom - lowestTop shortestHeight = min(self.height, zone.height) return ((1.0 * overlapAmount) / shortestHeight) > 0.25 else: return False
def getWordAtOffset(self, charOffset): word = None offset = 0 for word in self.words: nextOffset = offset + len(word.string) if nextOffset > charOffset: return [word, charOffset - offset] else: offset = nextOffset
return [word, offset]
class TextZone(Zone): """Represents Accessibility_Text that is a portion of a single horizontal line."""
def __init__(self, accessible, startOffset, string, x, y, width, height): """Creates a new Zone, which is a horizontal region of text.
Arguments: - accessible: the Accessible associated with this Zone - startOffset: the index of the char in the Accessibility_Text interface where this Zone starts - string: the string being displayed for this Zone - extents: x, y, width, height in screen coordinates """
Zone.__init__(self, accessible, string, x, y, width, height) self.startOffset = startOffset
# If the accessible for this TextZone is multiline, we will # keep track of the next and previous lines. # self.previousLineZone = None self.nextLineZone = None
def __getattr__(self, attr): """Used for lazily determining the words in a Zone. The words will either be all whitespace (interword boundaries) or actual words. To determine if a Word is whitespace, use word.string.isspace()
Arguments: - attr: a string indicating the attribute name to retrieve
Returns the value of the given attribute. """
if attr == "words": text = self.accessible.text self.words = [] wordIndex = 0 offset = self.startOffset for string in whitespace_re.split(self.string): if len(string): endOffset = offset + len(string) [x, y, width, height] = text.getRangeExtents( offset, endOffset, 0) word = Word(self, wordIndex, offset, string, x, y, width, height) self.words.append(word) wordIndex += 1 offset = endOffset
return self.words
elif attr.startswith('__') and attr.endswith('__'): raise AttributeError, attr else: return self.__dict__[attr]
class Line: """A Line is a single line across a window and is composed of Zones."""
def __init__(self, index, zones): """Creates a new Line, which is a horizontal region of text.
Arguments: - index: the index of this Line in the window - zones: the Zones that make up this line """
self.index = index self.zones = zones
bounds = None self.string = "" for zone in self.zones: if not bounds: bounds = [zone.x, zone.y, zone.x + zone.width, zone.y + zone.height] else: bounds[0] = min(bounds[0], zone.x) bounds[1] = min(bounds[1], zone.y) bounds[2] = max(bounds[2], zone.x + zone.width) bounds[3] = max(bounds[3], zone.y + zone.height) if len(zone.string): if len(self.string): self.string += " " self.string += zone.string
if not bounds: bounds = [-1, -1, -1, -1]
self.x = bounds[0] self.y = bounds[1] self.width = bounds[2] - bounds[0] self.height = bounds[3] - bounds[1]
self.brailleRegions = None
def getBrailleRegions(self): if not self.brailleRegions: self.brailleRegions = [] brailleOffset = 0 for zone in self.zones: if (zone.accessible.role == rolenames.ROLE_TEXT) \ or (zone.accessible.role == rolenames.ROLE_PASSWORD_TEXT) \ or (zone.accessible.role == rolenames.ROLE_TERMINAL): region = braille.ReviewText(zone.accessible, zone.string, zone.startOffset, zone) else: region = braille.ReviewComponent(zone.accessible, zone.string, 0, # cursor offset zone) if len(self.brailleRegions): pad = braille.Region(" ") pad.brailleOffset = brailleOffset self.brailleRegions.append(pad) brailleOffset += 1
zone.brailleRegion = region region.brailleOffset = brailleOffset self.brailleRegions.append(region)
brailleOffset += len(region.string)
if len(self.brailleRegions): pad = braille.Region(" ") pad.brailleOffset = brailleOffset self.brailleRegions.append(pad) brailleOffset += 1 eol = braille.Region("$l") eol.brailleOffset = brailleOffset self.brailleRegions.append(eol)
return self.brailleRegions
class Context: """Information regarding where a user happens to be exploring right now. """
ZONE = 0 CHAR = 1 WORD = 2 LINE = 3 # includes all zones on same line WINDOW = 4
WRAP_NONE = 0 WRAP_LINE = 1 << 0 WRAP_TOP_BOTTOM = 1 << 1 WRAP_ALL = (WRAP_LINE | WRAP_TOP_BOTTOM)
def __init__(self, lines, lineIndex, zoneIndex, wordIndex, charIndex): """Create a new Context that will be used for handling flat review mode.
Arguments: - lines: an array of arrays of Zones (see clusterZonesByLine) - lineIndex: index into lines - zoneIndex: index into lines[lineIndex].zones - wordIndex: index into lines[lineIndex].zones[zoneIndex].words - charIndex: index lines[lineIndex].zones[zoneIndex].words[wordIndex].chars """
self.lines = lines self.lineIndex = lineIndex self.zoneIndex = zoneIndex self.wordIndex = wordIndex self.charIndex = charIndex
# This is used to tell us where we should strive to move to # when going up and down lines to the closest character. # The targetChar is the character where we initially started # moving from, and does not change when one moves up or down # by line. # self.targetCharInfo = None
def _dumpCurrentState(self): print "line=%d, zone=%d, word=%d, char=%d" \ % (self.lineIndex, self.zoneIndex, self.wordIndex, self.zoneIndex)
zone = self.lines[self.lineIndex].zones[self.zoneIndex] text = zone.accessible.text
if not text: print " Not Accessibility_Text" return
print " getTextBeforeOffset: %d" % text.caretOffset [string, startOffset, endOffset] = text.getTextBeforeOffset( text.caretOffset, atspi.Accessibility.TEXT_BOUNDARY_WORD_START) print " WORD_START: start=%d end=%d string='%s'" \ % (startOffset, endOffset, string) [string, startOffset, endOffset] = text.getTextBeforeOffset( text.caretOffset, atspi.Accessibility.TEXT_BOUNDARY_WORD_END) print " WORD_END: start=%d end=%d string='%s'" \ % (startOffset, endOffset, string)
print " getTextAtOffset: %d" % text.caretOffset [string, startOffset, endOffset] = text.getTextAtOffset( text.caretOffset, atspi.Accessibility.TEXT_BOUNDARY_WORD_START) print " WORD_START: start=%d end=%d string='%s'" \ % (startOffset, endOffset, string) [string, startOffset, endOffset] = text.getTextAtOffset( text.caretOffset, atspi.Accessibility.TEXT_BOUNDARY_WORD_END) print " WORD_END: start=%d end=%d string='%s'" \ % (startOffset, endOffset, string)
print " getTextAfterOffset: %d" % text.caretOffset [string, startOffset, endOffset] = text.getTextAfterOffset( text.caretOffset, atspi.Accessibility.TEXT_BOUNDARY_WORD_START) print " WORD_START: start=%d end=%d string='%s'" \ % (startOffset, endOffset, string) [string, startOffset, endOffset] = text.getTextAfterOffset( text.caretOffset, atspi.Accessibility.TEXT_BOUNDARY_WORD_END) print " WORD_END: start=%d end=%d string='%s'" \ % (startOffset, endOffset, string)
def setCurrent(self, lineIndex, zoneIndex, wordIndex, charIndex): """Sets the current character of interest.
Arguments: - lineIndex: index into lines - zoneIndex: index into lines[lineIndex].zones - wordIndex: index into lines[lineIndex].zones[zoneIndex].words - charIndex: index lines[lineIndex].zones[zoneIndex].words[wordIndex].chars """
self.lineIndex = lineIndex self.zoneIndex = zoneIndex self.wordIndex = wordIndex self.charIndex = charIndex self.targetCharInfo = self.getCurrent(Context.CHAR)
#print "Current line=%d zone=%d word=%d char=%d" \ # % (lineIndex, zoneIndex, wordIndex, charIndex)
def clickCurrent(self, button=1): """Performs a mouse click on the current accessible."""
if (not self.lines) \ or (not self.lines[self.lineIndex].zones): return
[string, x, y, width, height] = self.getCurrent(Context.CHAR) try:
# We try to click to the left of center. This is to # handle toolkits that will offset the caret position to # the right if you click dead on center of a character. # eventsynthesizer.clickPoint(x, y + height/ 2, button) except: debug.printException(debug.LEVEL_SEVERE)
def getCurrentAccessible(self): """Returns the accessible associated with the current locus of interest. """
if (not self.lines) \ or (not self.lines[self.lineIndex].zones): return [None, -1, -1, -1, -1]
zone = self.lines[self.lineIndex].zones[self.zoneIndex]
return zone.accessible
def getCurrent(self, type=ZONE): """Gets the string, offset, and extent information for the current locus of interest.
Arguments: - type: one of ZONE, CHAR, WORD, LINE
Returns: [string, x, y, width, height] """
if (not self.lines) \ or (not self.lines[self.lineIndex].zones): return [None, -1, -1, -1, -1]
zone = self.lines[self.lineIndex].zones[self.zoneIndex]
if type == Context.ZONE: return [zone.string, zone.x, zone.y, zone.width, zone.height] elif type == Context.CHAR: if isinstance(zone, TextZone): words = zone.words if words: chars = zone.words[self.wordIndex].chars if chars: char = chars[self.charIndex] return [char.string, char.x, char.y, char.width, char.height] else: word = words[self.wordIndex] return [word.string, word.x, word.y, word.width, word.height] return self.getCurrent(Context.ZONE) elif type == Context.WORD: if isinstance(zone, TextZone): words = zone.words if words: word = words[self.wordIndex] return [word.string, word.x, word.y, word.width, word.height] return self.getCurrent(Context.ZONE) elif type == Context.LINE: line = self.lines[self.lineIndex] return [line.string, line.x, line.y, line.width, line.height] else: raise Exception("Invalid type: %d" % type)
def getCurrentBrailleRegions(self): """Gets the braille for the entire current line.
Returns [regions, regionWithFocus] """
if (not self.lines) \ or (not self.lines[self.lineIndex].zones): return [None, None]
regionWithFocus = None line = self.lines[self.lineIndex] regions = line.getBrailleRegions()
# Now find the current region and the current character offset # into that region. # for zone in line.zones: if zone.index == self.zoneIndex: regionWithFocus = zone.brailleRegion regionWithFocus.cursorOffset = 0 if zone.words: for wordIndex in range(0, self.wordIndex): regionWithFocus.cursorOffset += \ len(zone.words[wordIndex].string) regionWithFocus.cursorOffset += self.charIndex break
return [regions, regionWithFocus]
def goBegin(self, type=WINDOW): """Moves this context's locus of interest to the first char of the first relevant zone.
Arguments: - type: one of LINE or WINDOW
Returns True if the locus of interest actually changed. """
if type == Context.LINE: lineIndex = self.lineIndex elif type == Context.WINDOW: lineIndex = 0 else: raise Exception("Invalid type: %d" % type)
zoneIndex = 0 wordIndex = 0 charIndex = 0
moved = (self.lineIndex != lineIndex) \ or (self.zoneIndex != zoneIndex) \ or (self.wordIndex != wordIndex) \ or (self.charIndex != charIndex) \
if moved: self.lineIndex = lineIndex self.zoneIndex = zoneIndex self.wordIndex = wordIndex self.charIndex = charIndex self.targetCharInfo = self.getCurrent(Context.CHAR)
return moved
def goEnd(self, type=WINDOW): """Moves this context's locus of interest to the last char of the last relevant zone.
Arguments: - type: one of ZONE, LINE, or WINDOW
Returns True if the locus of interest actually changed. """
if type == Context.LINE: lineIndex = self.lineIndex elif type == Context.WINDOW: lineIndex = len(self.lines) - 1 else: raise Exception("Invalid type: %d" % type)
zoneIndex = len(self.lines[lineIndex].zones) - 1 zone = self.lines[lineIndex].zones[zoneIndex] if zone.words: wordIndex = len(zone.words) - 1 chars = zone.words[wordIndex].chars if chars: charIndex = len(chars) - 1 else: charIndex = 0 else: wordIndex = 0 charIndex = 0
moved = (self.lineIndex != lineIndex) \ or (self.zoneIndex != zoneIndex) \ or (self.wordIndex != wordIndex) \ or (self.charIndex != charIndex) \
if moved: self.lineIndex = lineIndex self.zoneIndex = zoneIndex self.wordIndex = wordIndex self.charIndex = charIndex self.targetCharInfo = self.getCurrent(Context.CHAR)
return moved
def goPrevious(self, type=ZONE, wrap=WRAP_ALL, omitWhitespace=True): """Moves this context's locus of interest to the first char of the previous type.
Arguments: - type: one of ZONE, CHAR, WORD, LINE - wrap: if True, will cross boundaries, including top and bottom; if False, will stop on boundaries.
Returns True if the locus of interest actually changed. """
moved = False
if type == Context.ZONE: if self.zoneIndex > 0: self.zoneIndex -= 1 self.wordIndex = 0 self.charIndex = 0 moved = True elif wrap & Context.WRAP_LINE: if self.lineIndex > 0: self.lineIndex -= 1 self.zoneIndex = len(self.lines[self.lineIndex].zones) - 1 self.wordIndex = 0 self.charIndex = 0 moved = True elif wrap & Context.WRAP_TOP_BOTTOM: self.lineIndex = len(self.lines) - 1 self.zoneIndex = len(self.lines[self.lineIndex].zones) - 1 self.wordIndex = 0 self.charIndex = 0 moved = True elif type == Context.CHAR: if self.charIndex > 0: self.charIndex -= 1 moved = True else: moved = self.goPrevious(Context.WORD, wrap, False) if moved: zone = self.lines[self.lineIndex].zones[self.zoneIndex] if zone.words: chars = zone.words[self.wordIndex].chars if chars: self.charIndex = len(chars) - 1 elif type == Context.WORD: zone = self.lines[self.lineIndex].zones[self.zoneIndex] accessible = zone.accessible lineIndex = self.lineIndex zoneIndex = self.zoneIndex wordIndex = self.wordIndex charIndex = self.charIndex
if self.wordIndex > 0: self.wordIndex -= 1 self.charIndex = 0 moved = True else: moved = self.goPrevious(Context.ZONE, wrap) if moved: zone = self.lines[self.lineIndex].zones[self.zoneIndex] if zone.words: self.wordIndex = len(zone.words) - 1
# If we landed on a whitespace word or something with no words, # we might need to move some more. # zone = self.lines[self.lineIndex].zones[self.zoneIndex] if omitWhitespace \ and moved \ and ((len(zone.words) == 0) \ or zone.words[self.wordIndex].string.isspace()):
# If we're on whitespace in the same zone, then let's # try to move on. If not, we've definitely moved # across accessibles. If that's the case, let's try # to find the first 'real' word in the accessible. # If we cannot, then we're just stuck on an accessible # with no words and we should do our best to announce # this to the user (e.g., "whitespace" or "blank"). # if zone.accessible == accessible: moved = self.goPrevious(Context.WORD, wrap) else: wordIndex = self.wordIndex - 1 while wordIndex >= 0: if (not zone.words[wordIndex].string) \ or not len(zone.words[wordIndex].string) \ or zone.words[wordIndex].string.isspace(): wordIndex -= 1 else: break if wordIndex >= 0: self.wordIndex = wordIndex
if not moved: self.lineIndex = lineIndex self.zoneIndex = zoneIndex self.wordIndex = wordIndex self.charIndex = charIndex
elif type == Context.LINE: if wrap & Context.WRAP_LINE: if self.lineIndex > 0: self.lineIndex -= 1 self.zoneIndex = 0 self.wordIndex = 0 self.charIndex = 0 moved = True elif (wrap & Context.WRAP_TOP_BOTTOM) \ and (len(self.lines) != 1): self.lineIndex = len(self.lines) - 1 self.zoneIndex = 0 self.wordIndex = 0 self.charIndex = 0 moved = True else: raise Exception("Invalid type: %d" % type)
if moved and (type != Context.LINE): self.targetCharInfo = self.getCurrent(Context.CHAR)
return moved
def goNext(self, type=ZONE, wrap=WRAP_ALL, omitWhitespace=True): """Moves this context's locus of interest to first char of the next type.
Arguments: - type: one of ZONE, CHAR, WORD, LINE - wrap: if True, will cross boundaries, including top and bottom; if False, will stop on boundaries. """
moved = False
if type == Context.ZONE: if self.zoneIndex < (len(self.lines[self.lineIndex].zones) - 1): self.zoneIndex += 1 self.wordIndex = 0 self.charIndex = 0 moved = True elif wrap & Context.WRAP_LINE: if self.lineIndex < (len(self.lines) - 1): self.lineIndex += 1 self.zoneIndex = 0 self.wordIndex = 0 self.charIndex = 0 moved = True elif wrap & Context.WRAP_TOP_BOTTOM: self.lineIndex = 0 self.zoneIndex = 0 self.wordIndex = 0 self.charIndex = 0 moved = True elif type == Context.CHAR: zone = self.lines[self.lineIndex].zones[self.zoneIndex] if zone.words: chars = zone.words[self.wordIndex].chars if chars: if self.charIndex < (len(chars) - 1): self.charIndex += 1 moved = True else: moved = self.goNext(Context.WORD, wrap, False) else: moved = self.goNext(Context.WORD, wrap) else: moved = self.goNext(Context.ZONE, wrap) elif type == Context.WORD: zone = self.lines[self.lineIndex].zones[self.zoneIndex] accessible = zone.accessible lineIndex = self.lineIndex zoneIndex = self.zoneIndex wordIndex = self.wordIndex charIndex = self.charIndex
if zone.words: if self.wordIndex < (len(zone.words) - 1): self.wordIndex += 1 self.charIndex = 0 moved = True else: moved = self.goNext(Context.ZONE, wrap) else: moved = self.goNext(Context.ZONE, wrap)
# If we landed on a whitespace word or something with no words, # we might need to move some more. # zone = self.lines[self.lineIndex].zones[self.zoneIndex] if omitWhitespace \ and moved \ and ((len(zone.words) == 0) \ or zone.words[self.wordIndex].string.isspace()):
# If we're on whitespace in the same zone, then let's # try to move on. If not, we've definitely moved # across accessibles. If that's the case, let's try # to find the first 'real' word in the accessible. # If we cannot, then we're just stuck on an accessible # with no words and we should do our best to announce # this to the user (e.g., "whitespace" or "blank"). # if zone.accessible == accessible: moved = self.goNext(Context.WORD, wrap) else: wordIndex = self.wordIndex + 1 while wordIndex < len(zone.words): if (not zone.words[wordIndex].string) \ or not len(zone.words[wordIndex].string) \ or zone.words[wordIndex].string.isspace(): wordIndex += 1 else: break if wordIndex < len(zone.words): self.wordIndex = wordIndex
if not moved: self.lineIndex = lineIndex self.zoneIndex = zoneIndex self.wordIndex = wordIndex self.charIndex = charIndex
elif type == Context.LINE: if wrap & Context.WRAP_LINE: if self.lineIndex < (len(self.lines) - 1): self.lineIndex += 1 self.zoneIndex = 0 self.wordIndex = 0 self.charIndex = 0 moved = True elif (wrap & Context.WRAP_TOP_BOTTOM) \ and (self.lineIndex != 0): self.lineIndex = 0 self.zoneIndex = 0 self.wordIndex = 0 self.charIndex = 0 moved = True else: raise Exception("Invalid type: %d" % type)
if moved and (type != Context.LINE): self.targetCharInfo = self.getCurrent(Context.CHAR)
return moved
def goAbove(self, type=LINE, wrap=WRAP_ALL): """Moves this context's locus of interest to first char of the type that's closest to and above the current locus of interest.
Arguments: - type: LINE - wrap: if True, will cross top/bottom boundaries; if False, will stop on top/bottom boundaries.
Returns: [string, startOffset, endOffset, x, y, width, height] """
moved = False if type == Context.CHAR: # We want to shoot for the closest character, which we've # saved away as self.targetCharInfo, which is the list # [string, x, y, width, height]. # if not self.targetCharInfo: self.targetCharInfo = self.getCurrent(Context.CHAR) target = self.targetCharInfo
[string, x, y, width, height] = target middleTargetX = x + (width / 2)
moved = self.goPrevious(Context.LINE, wrap) if moved: while True: [string, bx, by, bwidth, bheight] = \ self.getCurrent(Context.CHAR) if (bx + width) >= middleTargetX: break elif not self.goNext(Context.CHAR, Context.WRAP_NONE): break
# Moving around might have reset the current targetCharInfo, # so we reset it to our saved value. # self.targetCharInfo = target elif type == Context.LINE: return self.goPrevious(type, wrap) else: raise Exception("Invalid type: %d" % type)
return moved
def goBelow(self, type=LINE, wrap=WRAP_ALL): """Moves this context's locus of interest to the first char of the type that's closest to and below the current locus of interest.
Arguments: - type: one of WORD, LINE - wrap: if True, will cross top/bottom boundaries; if False, will stop on top/bottom boundaries.
Returns: [string, startOffset, endOffset, x, y, width, height] """
moved = False if type == Context.CHAR: # We want to shoot for the closest character, which we've # saved away as self.targetCharInfo, which is the list # [string, x, y, width, height]. # if not self.targetCharInfo: self.targetCharInfo = self.getCurrent(Context.CHAR) target = self.targetCharInfo
[string, x, y, width, height] = target middleTargetX = x + (width / 2)
moved = self.goNext(Context.LINE, wrap) if moved: while True: [string, bx, by, bwidth, bheight] = \ self.getCurrent(Context.CHAR) if (bx + width) >= middleTargetX: break elif not self.goNext(Context.CHAR, Context.WRAP_NONE): break
# Moving around might have reset the current targetCharInfo, # so we reset it to our saved value. # self.targetCharInfo = target elif type == Context.LINE: moved = self.goNext(type, wrap) else: raise Exception("Invalid type: %d" % type)
return moved
def visible(ax, ay, awidth, aheight, bx, by, bwidth, bheight): """Returns true if any portion of region 'a' is in region 'b' """ highestBottom = min(ay + aheight, by + bheight) lowestTop = max(ay, by)
leftMostRightEdge = min(ax + awidth, bx + bwidth) rightMostLeftEdge = max(ax, bx)
visible = False
if (lowestTop < highestBottom) \ and (rightMostLeftEdge < leftMostRightEdge): visible = True elif (aheight == 0): if (awidth == 0): visible = (lowestTop == highestBottom) \ and (leftMostRightEdge == rightMostLeftEdge) else: visible = leftMostRightEdge < rightMostLeftEdge elif (awidth == 0): visible = (lowestTop < highestBottom)
return visible
def clip(ax, ay, awidth, aheight, bx, by, bwidth, bheight): """Clips region 'a' by region 'b' and returns the new region as a list: [x, y, width, height]. """
x = max(ax, bx) x2 = min(ax + awidth, bx + bwidth) width = x2 - x
y = max(ay, by) y2 = min(ay + aheight, by + bheight) height = y2 - y
return [x, y, width, height]
def getZonesFromAccessible(accessible, cliprect): """Returns a list of Zones for the given accessible.
Arguments: - accessible: the accessible - cliprect: the extents that the Zones must fit inside. """
if not accessible.component: return []
# Get the component extents in screen coordinates. # extents = accessible.component.getExtents(0)
if not visible(extents.x, extents.y, extents.width, extents.height, cliprect.x, cliprect.y, cliprect.width, cliprect.height): return []
zones = []
debug.println( debug.LEVEL_FINEST, "flat_review.getZonesFromAccessible (name=%s role=%s)" \ % (accessible.name, accessible.role))
# Now see if there is any accessible text. If so, find new zones, # where each zone represents a line of this text object. When # creating the zone, only keep track of the text that is actually # showing on the screen. # if accessible.text: debug.println(debug.LEVEL_FINEST, " looking at text:")
text = accessible.text length = text.characterCount
offset = 0 lastEndOffset = -1 while offset < length:
[string, startOffset, endOffset] = text.getTextAtOffset( offset, atspi.Accessibility.TEXT_BOUNDARY_LINE_START)
debug.println(debug.LEVEL_FINEST, " line at %d is (start=%d end=%d): '%s'" \ % (offset, startOffset, endOffset, string))
# [[[WDW - HACK: well...gnome-terminal sometimes wants to # give us outrageous values back from getTextAtOffset # (see http://bugzilla.gnome.org/show_bug.cgi?id=343133), # so we try to handle it. Evolution does similar things.]]] # if (startOffset < 0) \ or (endOffset < 0) \ or (startOffset > offset) \ or (endOffset < offset) \ or (startOffset > endOffset) \ or (abs(endOffset - startOffset) > 666e3): debug.println(debug.LEVEL_WARNING, "flat_review:getZonesFromAccessible detected "\ "garbage from getTextAtOffset for accessible "\ "name='%s' role'='%s': offset used=%d, "\ "start/end offset returned=(%d,%d), string='%s'"\ % (accessible.name, accessible.role, offset, startOffset, endOffset, string)) break
# [[[WDW - HACK: this is here because getTextAtOffset # tends not to be implemented consistently across toolkits. # Sometimes it behaves properly (i.e., giving us an endOffset # that is the beginning of the next line), sometimes it # doesn't (e.g., giving us an endOffset that is the end of # the current line). So...we hack. The whole 'max' deal # is to account for lines that might be a brazillion lines # long.]]] # if endOffset == lastEndOffset: offset = max(offset + 1, lastEndOffset + 1) lastEndOffset = endOffset continue
lastEndOffset = endOffset
[x, y, width, height] = text.getRangeExtents(startOffset, endOffset, 0)
offset = endOffset previousLineZone = None
if visible(x, y, width, height, cliprect.x, cliprect.y, cliprect.width, cliprect.height):
clipping = clip(x, y, width, height, cliprect.x, cliprect.y, cliprect.width, cliprect.height)
# [[[TODO: WDW - HACK it would be nice to clip the # the text by what is really showing on the screen, # but this seems to hang Orca and the client. Logged # as bugzilla bug 319770.]]] # #ranges = text.getBoundedRanges(\ # clipping[0], # clipping[1], # clipping[2], # clipping[3], # 0, # atspi.Accessibility.TEXT_CLIP_BOTH, # atspi.Accessibility.TEXT_CLIP_BOTH) # #print #print "HERE!" #for range in ranges: # print range.startOffset # print range.endOffset # print range.content
zone = TextZone(accessible, startOffset, string, clipping[0], clipping[1], clipping[2], clipping[3])
if previousLineZone: previousLineZone.nextLineZone = zone zone.previousLineZone = previousLineZone zone.nextLineZone = None
zones.append(zone)
previousLineZone = zone
elif len(zones): # We'll break out of searching all the text - the idea # here is that we'll at least try to optimize for when # we gone below the visible clipping area. # # [[[TODO: WDW - would be nice to optimize this better. # for example, perhaps we can assume the caret will always # be visible, and we can start our text search from there. # Logged as bugzilla bug 319771.]]] # break
# We might have a zero length text area. In that case, well, # lets hack... # if len(zones) == 0: if (accessible.role == rolenames.ROLE_TEXT) \ or ((accessible.role == rolenames.ROLE_PASSWORD_TEXT)): zones.append(TextZone(accessible, 0, "", extents.x, extents.y, extents.width, extents.height))
# We really want the accessible text information. But, if we have # an image, and it has a description, we can fall back on it. # if (len(zones) == 0) \ and accessible.image \ and accessible.image.imageDescription \ and len(accessible.image.imageDescription):
[x, y] = accessible.image.getImagePosition(0) [width, height] = accessible.image.getImageSize()
if (width != 0) and (height != 0) \ and visible(x, y, width, height, cliprect.x, cliprect.y, cliprect.width, cliprect.height):
clipping = clip(x, y, width, height, cliprect.x, cliprect.y, cliprect.width, cliprect.height)
if (clipping[2] != 0) or (clipping[3] != 0): zones.append(Zone(accessible, accessible.image.imageDescription, clipping[0], clipping[1], clipping[2], clipping[3]))
# Well...darn. Maybe we didn't get anything of use, but we certainly # know there's something there. If that's the case, we'll just use # the component extents and the name or description of the accessible. # if len(zones) == 0: clipping = clip(extents.x, extents.y, extents.width, extents.height, cliprect.x, cliprect.y, cliprect.width, cliprect.height) if accessible.name and len(accessible.name): string = accessible.name elif accessible.description and len(accessible.description): string = accessible.description else: string = ""
if string == "": # [[[TODO: WDW - ooohhhh....this is going to be a headache. # We want to synthesize a string for objects that are there, # but have no text. For example, scroll bars. The rub is # that we will sometimes want to do different strings for # speech and braille (e.g., checkbox cells in tables - the # speech will say "checkbox checked" and the braille will # display "checkbox <x>". Yikes. That may be a challenge. # So...for now we punt on table cells. Logged as bugzilla # bug 319772.]]] # if accessible.role != rolenames.ROLE_TABLE_CELL: string = accessible.role
if len(string) and ((clipping[2] != 0) or (clipping[3] != 0)): zones.append(Zone(accessible, string, clipping[0], clipping[1], clipping[2], clipping[3]))
return zones
def getShowingDescendants(parent): """Given a parent that manages its descendants, return a list of Accessible children that are actually showing. This algorithm was inspired a little by the srw_elements_from_accessible logic in Gnopernicus, and makes the assumption that the children of an object that manages its descendants are arranged in a row and column format. """
if (not parent) or (not parent.component): return []
# A minimal chunk to jump around should we not really know where we # are going. # GRID_SIZE = 7
descendants = []
parentExtents = parent.component.getExtents(0)
# [[[TODO: WDW - HACK related to GAIL bug where table column headers # seem to be ignored: http://bugzilla.gnome.org/show_bug.cgi?id=325809. # The problem is that this causes getAccessibleAtPoint to return the # cell effectively below the real cell at a given point, making a mess # of everything. So...we just manually add in showing headers for now. # The remainder of the logic below accidentally accounts for this offset, # yet it should also work when bug 325809 is fixed.]]] # table = parent.table if table: for i in range(0, table.nColumns): obj = table.getColumnHeader(i) if obj: header = atspi.Accessible.makeAccessible(obj) extents = header.extents if header.state.count(atspi.Accessibility.STATE_SHOWING) \ and (extents.x >= 0) and (extents.y >= 0) \ and (extents.width > 0) and (extents.height > 0) \ and visible(extents.x, extents.y, extents.width, extents.height, parentExtents.x, parentExtents.y, parentExtents.width, parentExtents.height): descendants.append(header)
# This algorithm goes left to right, top to bottom while attempting # to do *some* optimization over queries. It could definitely be # improved. # currentY = parentExtents.y while currentY < (parentExtents.y + parentExtents.height): currentX = parentExtents.x minHeight = sys.maxint while currentX < (parentExtents.x + parentExtents.width): obj = parent.component.getAccessibleAtPoint(currentX, currentY, 0) if obj: child = atspi.Accessible.makeAccessible(obj) extents = child.extents if extents.x >= 0 and extents.y >= 0: newX = extents.x + extents.width minHeight = min(minHeight, extents.height) if not descendants.count(child): descendants.append(child) else: newX = currentX + GRID_SIZE else: newX = currentX + GRID_SIZE if newX <= currentX: currentX += GRID_SIZE else: currentX = newX if minHeight == sys.maxint: minHeight = GRID_SIZE currentY += minHeight
return descendants
def getShowingZones(root): """Returns a list of all interesting, non-intersecting, regions that are drawn on the screen. Each element of the list is the Accessible object associated with a given region. The term 'zone' here is inherited from OCR algorithms and techniques.
The Zones are returned in no particular order.
Arguments: - root: the Accessible object to traverse
Returns: a list of Zones under the specified object """
if not root: return []
# If we're at a leaf node, then we've got a good one on our hands. # if root.childCount <= 0: return getZonesFromAccessible(root, root.extents)
# We'll stop at various objects because, while they do have # children, we logically think of them as one region on the # screen. [[[TODO: WDW - HACK stopping at menu bars for now # because their menu items tell us they are showing even though # they are not showing. Until I can figure out a reliable way to # get past these lies, I'm going to ignore them.]]] # if (root.parent and (root.parent.role == rolenames.ROLE_MENU_BAR)) \ or (root.role == rolenames.ROLE_COMBO_BOX) \ or (root.role == rolenames.ROLE_TEXT): return getZonesFromAccessible(root, root.extents)
# Otherwise, dig deeper. # objlist = []
# We'll include page tabs: while they are parents, their extents do # not contain their children. [[[TODO: WDW - need to consider all # parents, especially those that implement accessible text. Logged # as bugzilla bug 319773.]]] # if root.role == rolenames.ROLE_PAGE_TAB: objlist.extend(getZonesFromAccessible(root, root.extents))
if root.state.count(atspi.Accessibility.STATE_MANAGES_DESCENDANTS) \ and (root.childCount > 50): for child in getShowingDescendants(root): objlist.extend(getShowingZones(child)) else: for i in range(0, root.childCount): child = root.child(i) if child == root: debug.println(debug.LEVEL_WARNING, "flat_review.getShowingZones: " + "WARNING CHILD == PARENT!!!") elif not child: debug.println(debug.LEVEL_WARNING, "flat_review.getShowingZones: " + "WARNING CHILD IS NONE!!!") elif child.parent != root: debug.println(debug.LEVEL_WARNING, "flat_review.getShowingZones: " + "WARNING CHILD.PARENT != PARENT!!!") elif child.state.count(atspi.Accessibility.STATE_SHOWING): objlist.extend(getShowingZones(child))
return objlist
def clusterZonesByLine(zones): """Given a list of interesting accessible objects (the Zones), returns a list of lines in order from the top to bottom, where each line is a list of accessible objects in order from left to right. """
if len(zones) == 0: return []
# Sort the zones and also find the top most zone - we'll bias # the clustering to the top of the window. That is, if an # object can be part of multiple clusters, for now it will # become a part of the top most cluster. # numZones = len(zones) for i in range(0, numZones): for j in range(0, numZones - 1 - i): a = zones[j] b = zones[j + 1] if b.y < a.y: zones[j] = b zones[j + 1] = a
# Now we cluster the zones. We create the clusters on the # fly, adding a zone to an existing cluster only if it's # rectangle horizontally overlaps all other zones in the # cluster. # lineClusters = [] for clusterCandidate in zones: addedToCluster = False for lineCluster in lineClusters: inCluster = True for zone in lineCluster: if not zone.onSameLine(clusterCandidate): inCluster = False break if inCluster: # Add to cluster based on the x position. # i = 0 while i < len(lineCluster): zone = lineCluster[i] if clusterCandidate.x < zone.x: break else: i += 1 lineCluster.insert(i, clusterCandidate) addedToCluster = True break if not addedToCluster: lineClusters.append([clusterCandidate])
# Now, adjust all the indeces. # lines = [] lineIndex = 0 for lineCluster in lineClusters: lines.append(Line(lineIndex, lineCluster)) zoneIndex = 0 for zone in lineCluster: zone.line = lines[lineIndex] zone.index = zoneIndex zoneIndex += 1 lineIndex += 1
return lines
|