%PDF- %PDF-
| Direktori : /lib/python3/dist-packages/orca/scripts/toolkits/Gecko/ |
| Current File : //lib/python3/dist-packages/orca/scripts/toolkits/Gecko/script_utilities.py |
# Orca
#
# Copyright 2010 Joanmarie Diggs.
# Copyright 2014-2015 Igalia, S.L.
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2.1 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
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this library; if not, write to the
# Free Software Foundation, Inc., Franklin Street, Fifth Floor,
# Boston MA 02110-1301 USA.
"""Commonly-required utility methods needed by -- and potentially
customized by -- application and toolkit scripts. They have
been pulled out from the scripts because certain scripts had
gotten way too large as a result of including these methods."""
__id__ = "$Id$"
__version__ = "$Revision$"
__date__ = "$Date$"
__copyright__ = "Copyright (c) 2010 Joanmarie Diggs." \
"Copyright (c) 2014-2015 Igalia, S.L."
__license__ = "LGPL"
import gi
gi.require_version("Atspi", "2.0")
import re
import time
from orca import debug
from orca import focus_manager
from orca import orca_state
from orca.scripts import web
from orca.ax_document import AXDocument
from orca.ax_object import AXObject
from orca.ax_utilities import AXUtilities
class Utilities(web.Utilities):
def __init__(self, script):
super().__init__(script)
self._lastAutoTextObjectEvent = None
self._lastAutoTextInputEvent = None
self._lastAutoTextEventTime = 0
def isLayoutOnly(self, obj):
if super().isLayoutOnly(obj):
return True
if AXUtilities.is_tool_bar(obj) and AXObject.get_child_count(obj):
return AXUtilities.is_page_tab_list(AXObject.get_child(obj, 0))
return False
def isSameObject(self, obj1, obj2, comparePaths=False, ignoreNames=False,
ignoreDescriptions=True):
if super().isSameObject(obj1, obj2, comparePaths, ignoreNames, ignoreDescriptions):
return True
roles = self._topLevelRoles()
if not (AXObject.get_role(obj1) in roles and AXObject.get_role(obj2) in roles):
return False
rv = AXObject.get_name(obj1) == AXObject.get_name(obj2)
tokens = ["GECKO: Treating", obj1, "and", obj2, "as same object:", rv]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return rv
def getOnScreenObjects(self, root, extents=None):
objects = super().getOnScreenObjects(root, extents)
# For things like Thunderbird's "Select columns to display" button
if AXUtilities.is_tree_table(root) and AXObject.get_child_count(root):
def isExtra(x):
return not AXUtilities.is_column_header(x)
child = AXObject.get_child(root, 0)
objects.extend([x for x in AXObject.iter_children(child, isExtra)])
return objects
def isEditableMessage(self, obj):
"""Returns True if this is an editable message."""
if not AXUtilities.is_editable(obj):
return False
document = self.getDocumentForObject(obj)
if AXUtilities.is_editable(document):
tokens = ["GECKO:", obj, "is in an editable document:", document]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return True
tokens = ["GECKO: Editable", obj, "not in an editable document"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return False
def isNotRealDocument(self, obj):
name = AXObject.get_name(obj)
if name.startswith("moz-extension"):
return True
if "self-repair.mozilla.org" in name:
return True
uri = AXDocument.get_uri(obj)
if uri.startswith("moz-extension"):
return True
if not uri and "pixels" in name:
return True
return False
def treatAsEntry(self, obj):
if not obj or self.inDocumentContent(obj):
return super().treatAsEntry(obj)
# Firefox seems to have turned its accessible location widget into a
# childless editable combobox.
if not AXUtilities.is_combo_box(obj):
return False
if AXObject.get_child_count(obj):
return False
if not AXUtilities.is_focused(obj):
return False
if not AXObject.supports_editable_text(obj):
return False
tokens = ["GECKO: Treating", obj, "as entry"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return True
def _isQuickFind(self, obj):
if not obj or self.inDocumentContent(obj):
return False
if obj == self._findContainer:
return True
if not AXUtilities.is_tool_bar(obj):
return False
# TODO: This would be far easier if Gecko gave us an object attribute to look for....
if len(AXUtilities.find_all_entries(obj)) != 1:
tokens = ["GECKO:", obj, "not believed to be quick-find container (entry count)"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return False
if len(AXUtilities.find_all_push_buttons(obj)) != 1:
tokens = ["GECKO:", obj, "not believed to be quick-find container (button count)"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return False
tokens = ["GECKO:", obj, "believed to be quick-find container (accessibility tree)"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
self._findContainer = obj
return True
def isFindContainer(self, obj):
if not obj or self.inDocumentContent(obj):
return False
if obj == self._findContainer:
return True
if not AXUtilities.is_tool_bar(obj):
return False
result = self.getFindResultsCount(obj)
if result:
tokens = ["GECKO:", obj, "believed to be find-in-page container (", result, ")"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
self._findContainer = obj
return True
# TODO: This would be far easier if Gecko gave us an object attribute to look for....
if len(AXUtilities.find_all_entries(obj)) != 1:
tokens = ["GECKO:", obj, "not believed to be find-in-page container (entry count)"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return False
if len(AXUtilities.find_all_push_buttons(obj)) < 5:
tokens = ["GECKO:", obj, "not believed to be find-in-page container (button count)"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return False
tokens = ["GECKO:", obj, "believed to be find-in-page container (accessibility tree)"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
self._findContainer = obj
return True
def inFindContainer(self, obj=None):
if not obj:
obj = focus_manager.getManager().get_locus_of_focus()
if not obj or self.inDocumentContent(obj):
return False
if not (AXUtilities.is_entry(obj) or AXUtilities.is_push_button(obj)):
return False
toolbar = AXObject.find_ancestor(obj, AXUtilities.is_tool_bar)
result = self.isFindContainer(toolbar)
if result:
tokens = ["GECKO:", obj, "believed to be find-in-page widget (toolbar)"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return True
if self._isQuickFind(toolbar):
tokens = ["GECKO:", obj, "believed to be find-in-page widget (quick find)"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return True
return False
def getFindResultsCount(self, root=None):
root = root or self._findContainer
if not root:
return ""
def isMatch(x):
return len(re.findall(r"\d+", AXObject.get_name(x))) == 2
labels = AXUtilities.find_all_labels(root, isMatch)
if len(labels) != 1:
return ""
label = labels[0]
AXObject.clear_cache(label, False, "Ensuring we have correct name for find results.")
return AXObject.get_name(label)
def isAutoTextEvent(self, event):
if not super().isAutoTextEvent(event):
return False
if self.inDocumentContent(event.source):
return True
if self.treatAsDuplicateEvent(self._lastAutoTextObjectEvent, event) \
and time.time() - self._lastAutoTextEventTime < 0.5 \
and orca_state.lastInputEvent.isReleaseFor(self._lastAutoTextInputEvent):
msg = "GECKO: Event believed to be duplicate auto text event."
debug.printMessage(debug.LEVEL_INFO, msg, True)
return False
self._lastAutoTextObjectEvent = event
self._lastAutoTextInputEvent = orca_state.lastInputEvent
self._lastAutoTextEventTime = time.time()
return True
def localizeTextAttribute(self, key, value):
value = value.replace("-moz-", "")
return super().localizeTextAttribute(key, value)
def unrelatedLabels(self, root, onlyShowing=True, minimumWords=3):
return super().unrelatedLabels(root, onlyShowing, minimumWords=1)