mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
298 lines
12 KiB
C
298 lines
12 KiB
C
|
/* ***** BEGIN LICENSE BLOCK *****
|
||
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||
|
*
|
||
|
* The contents of this file are subject to the Mozilla Public License Version
|
||
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
||
|
* the License. You may obtain a copy of the License at
|
||
|
* http://www.mozilla.org/MPL/
|
||
|
*
|
||
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
||
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
||
|
* for the specific language governing rights and limitations under the
|
||
|
* License.
|
||
|
*
|
||
|
* The Original Code is mozilla.org code.
|
||
|
*
|
||
|
* The Initial Developer of the Original Code is
|
||
|
* Mozilla Foundation.
|
||
|
* Portions created by the Initial Developer are Copyright (C) 2011
|
||
|
* the Initial Developer. All Rights Reserved.
|
||
|
*
|
||
|
* Contributor(s):
|
||
|
* Alexander Surkov <surkov.alexander@gmail.com> (original author)
|
||
|
*
|
||
|
* Alternatively, the contents of this file may be used under the terms of
|
||
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
||
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
||
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
||
|
* of those above. If you wish to allow use of your version of this file only
|
||
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
||
|
* use your version of this file under the terms of the MPL, indicate your
|
||
|
* decision by deleting the provisions above and replace them with the notice
|
||
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
||
|
* the provisions above, a recipient may use your version of this file under
|
||
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
||
|
*
|
||
|
* ***** END LICENSE BLOCK ***** */
|
||
|
|
||
|
#ifndef mozilla_a11y_FocusManager_h_
|
||
|
#define mozilla_a11y_FocusManager_h_
|
||
|
|
||
|
#include "nsAutoPtr.h"
|
||
|
#include "mozilla/dom/Element.h"
|
||
|
|
||
|
class AccEvent;
|
||
|
class nsAccessible;
|
||
|
class nsDocAccessible;
|
||
|
|
||
|
namespace mozilla {
|
||
|
namespace a11y {
|
||
|
|
||
|
/**
|
||
|
* Manage the accessible focus. Used to fire and process accessible events.
|
||
|
*/
|
||
|
class FocusManager
|
||
|
{
|
||
|
public:
|
||
|
virtual ~FocusManager();
|
||
|
|
||
|
/**
|
||
|
* Return a focused accessible.
|
||
|
*/
|
||
|
nsAccessible* FocusedAccessible() const;
|
||
|
|
||
|
/**
|
||
|
* Return true if given accessible is focused.
|
||
|
*/
|
||
|
bool IsFocused(const nsAccessible* aAccessible) const;
|
||
|
|
||
|
/**
|
||
|
* Return true if the given accessible is an active item, i.e. an item that
|
||
|
* is current within the active widget.
|
||
|
*/
|
||
|
inline bool IsActiveItem(const nsAccessible* aAccessible)
|
||
|
{ return aAccessible == mActiveItem; }
|
||
|
|
||
|
/**
|
||
|
* Return true if given DOM node has DOM focus.
|
||
|
*/
|
||
|
inline bool HasDOMFocus(const nsINode* aNode) const
|
||
|
{ return aNode == FocusedDOMNode(); }
|
||
|
|
||
|
/**
|
||
|
* Return true if focused accessible is within the given container.
|
||
|
*/
|
||
|
bool IsFocusWithin(const nsAccessible* aContainer) const;
|
||
|
|
||
|
/**
|
||
|
* Return whether the given accessible is focused or contains the focus or
|
||
|
* contained by focused accessible.
|
||
|
*/
|
||
|
enum FocusDisposition {
|
||
|
eNone,
|
||
|
eFocused,
|
||
|
eContainsFocus,
|
||
|
eContainedByFocus
|
||
|
};
|
||
|
FocusDisposition IsInOrContainsFocus(const nsAccessible* aAccessible) const;
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////////
|
||
|
// Focus notifications and processing (don't use until you know what you do).
|
||
|
|
||
|
/**
|
||
|
* Called when DOM focus event is fired.
|
||
|
*/
|
||
|
void NotifyOfDOMFocus(nsISupports* aTarget);
|
||
|
|
||
|
/**
|
||
|
* Called when DOM blur event is fired.
|
||
|
*/
|
||
|
void NotifyOfDOMBlur(nsISupports* aTarget);
|
||
|
|
||
|
/**
|
||
|
* Called when active item is changed. Note: must be called when accessible
|
||
|
* tree is up to date.
|
||
|
*/
|
||
|
void ActiveItemChanged(nsAccessible* aItem, bool aCheckIfActive = true);
|
||
|
|
||
|
/**
|
||
|
* Dispatch delayed focus event for the current focus accessible.
|
||
|
*/
|
||
|
void ForceFocusEvent();
|
||
|
|
||
|
/**
|
||
|
* Dispatch delayed focus event for the given target.
|
||
|
*/
|
||
|
void DispatchFocusEvent(nsDocAccessible* aDocument, nsAccessible* aTarget);
|
||
|
|
||
|
/**
|
||
|
* Process DOM focus notification.
|
||
|
*/
|
||
|
void ProcessDOMFocus(nsINode* aTarget);
|
||
|
|
||
|
/**
|
||
|
* Process the delayed accessible event.
|
||
|
* do.
|
||
|
*/
|
||
|
void ProcessFocusEvent(AccEvent* aEvent);
|
||
|
|
||
|
protected:
|
||
|
FocusManager();
|
||
|
|
||
|
private:
|
||
|
FocusManager(const FocusManager&);
|
||
|
FocusManager& operator =(const FocusManager&);
|
||
|
|
||
|
/**
|
||
|
* Return DOM node having DOM focus.
|
||
|
*/
|
||
|
inline nsINode* FocusedDOMNode() const
|
||
|
{
|
||
|
nsINode* focusedNode = FocusedDOMElm();
|
||
|
if (focusedNode)
|
||
|
return focusedNode;
|
||
|
return FocusedDOMDocument();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return DOM element having DOM focus.
|
||
|
*/
|
||
|
nsIContent* FocusedDOMElm() const;
|
||
|
|
||
|
/**
|
||
|
* Return DOM document having DOM focus.
|
||
|
*/
|
||
|
nsIDocument* FocusedDOMDocument() const;
|
||
|
|
||
|
private:
|
||
|
nsRefPtr<nsAccessible> mActiveItem;
|
||
|
nsRefPtr<nsAccessible> mActiveARIAMenubar;
|
||
|
};
|
||
|
|
||
|
} // namespace a11y
|
||
|
} // namespace mozilla
|
||
|
|
||
|
|
||
|
//#define A11YDEBUG_FOCUS
|
||
|
|
||
|
#ifdef A11YDEBUG_FOCUS
|
||
|
|
||
|
// Util macros (don't use them directly)
|
||
|
#define A11YDEBUG_FOCUS_STARTBLOCK \
|
||
|
printf(" {\n ");
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_ENDBLOCK \
|
||
|
printf("\n }\n");
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_BLOCKOFFSET \
|
||
|
printf(" ");
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_LOG_TIME \
|
||
|
{ \
|
||
|
PRIntervalTime time = PR_IntervalNow(); \
|
||
|
PRUint32 mins = (PR_IntervalToSeconds(time) / 60) % 60; \
|
||
|
PRUint32 secs = PR_IntervalToSeconds(time) % 60; \
|
||
|
PRUint32 msecs = PR_IntervalToMilliseconds(time) % 1000; \
|
||
|
printf("Time: %2d:%2d.%3d\n", mins, secs, msecs); \
|
||
|
}
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_LOG_DOMNODE(aNode) \
|
||
|
if (aNode) { \
|
||
|
if (aNode->IsElement()) { \
|
||
|
dom::Element* targetElm = aNode->AsElement(); \
|
||
|
nsCAutoString tag; \
|
||
|
targetElm->Tag()->ToUTF8String(tag); \
|
||
|
nsCAutoString id; \
|
||
|
nsIAtom* atomid = targetElm->GetID(); \
|
||
|
if (atomid) \
|
||
|
atomid->ToUTF8String(id); \
|
||
|
printf("element %s@id='%s': %p", tag.get(), id.get(), (void*)aNode); \
|
||
|
} else if (aNode->IsNodeOfType(nsINode::eDOCUMENT)) { \
|
||
|
nsCOMPtr<nsIDocument> document = do_QueryInterface(aNode); \
|
||
|
nsIURI* uri = document->GetDocumentURI(); \
|
||
|
nsCAutoString spec; \
|
||
|
uri->GetSpec(spec); \
|
||
|
printf("document: %p; uri: %s", (void*)aNode, spec.get()); \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_LOG_ACCESSIBLE(aAccessible) \
|
||
|
printf("accessible: %p; ", (void*)aAccessible); \
|
||
|
if (aAccessible) { \
|
||
|
nsAutoString role; \
|
||
|
GetAccService()->GetStringRole(aAccessible->Role(), role); \
|
||
|
nsAutoString name; \
|
||
|
aAccessible->GetName(name); \
|
||
|
printf(" role: %s, name: %s; ", NS_ConvertUTF16toUTF8(role).get(), \
|
||
|
NS_ConvertUTF16toUTF8(name).get()); \
|
||
|
A11YDEBUG_FOCUS_LOG_DOMNODE(aAccessible->GetNode()) \
|
||
|
}
|
||
|
|
||
|
// Public macros
|
||
|
#define A11YDEBUG_FOCUS_LOG_DOMTARGET(aMsg, aTarget) \
|
||
|
A11YDEBUG_FOCUS_STARTBLOCK \
|
||
|
printf(aMsg "\n"); \
|
||
|
if (aTarget) { \
|
||
|
A11YDEBUG_FOCUS_BLOCKOFFSET \
|
||
|
A11YDEBUG_FOCUS_LOG_DOMNODE(aTarget) \
|
||
|
} \
|
||
|
A11YDEBUG_FOCUS_ENDBLOCK
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_LOG_ACCTARGET(aMsg, aTarget) \
|
||
|
A11YDEBUG_FOCUS_STARTBLOCK \
|
||
|
printf(aMsg "\n"); \
|
||
|
A11YDEBUG_FOCUS_BLOCKOFFSET \
|
||
|
A11YDEBUG_FOCUS_LOG_ACCESSIBLE(aTarget) \
|
||
|
A11YDEBUG_FOCUS_ENDBLOCK
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_LOG_WIDGET(aMsg, aWidget) \
|
||
|
A11YDEBUG_FOCUS_STARTBLOCK \
|
||
|
printf(aMsg "\n"); \
|
||
|
A11YDEBUG_FOCUS_BLOCKOFFSET \
|
||
|
A11YDEBUG_FOCUS_LOG_ACCESSIBLE(aWidget) \
|
||
|
printf("; widget is active: %s, has operable items: %s", \
|
||
|
(aWidget && aWidget->IsActiveWidget() ? "true" : "false"), \
|
||
|
(aWidget && aWidget->AreItemsOperable() ? "true" : "false")); \
|
||
|
A11YDEBUG_FOCUS_ENDBLOCK
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_NOTIFICATION_SUPPORTSTARGET(aMsg, aTargetMsg, aTarget) \
|
||
|
printf("\nA11Y FOCUS: " aMsg ". "); \
|
||
|
A11YDEBUG_FOCUS_LOG_TIME \
|
||
|
if (aTarget) { \
|
||
|
A11YDEBUG_FOCUS_STARTBLOCK \
|
||
|
printf(aTargetMsg "\n"); \
|
||
|
A11YDEBUG_FOCUS_BLOCKOFFSET \
|
||
|
nsCOMPtr<nsINode> targetNode(do_QueryInterface(aTarget)); \
|
||
|
if (targetNode) { \
|
||
|
A11YDEBUG_FOCUS_LOG_DOMNODE(targetNode) \
|
||
|
} else { \
|
||
|
printf("window: %p", (void*)aTarget); \
|
||
|
} \
|
||
|
A11YDEBUG_FOCUS_ENDBLOCK \
|
||
|
}
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_NOTIFICATION_DOMTARGET(aMsg, aTargetMsg, aTarget) \
|
||
|
printf("\nA11Y FOCUS: " aMsg ". "); \
|
||
|
A11YDEBUG_FOCUS_LOG_TIME \
|
||
|
A11YDEBUG_FOCUS_LOG_DOMTARGET(aTargetMsg, aTarget)
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_NOTIFICATION_ACCTARGET(aMsg, aTargetMsg, aTarget) \
|
||
|
printf("\nA11Y FOCUS: " aMsg ". "); \
|
||
|
A11YDEBUG_FOCUS_LOG_TIME \
|
||
|
A11YDEBUG_FOCUS_LOG_ACCTARGET(aTargetMsg, aTarget)
|
||
|
|
||
|
#define A11YDEBUG_FOCUS_ACTIVEITEMCHANGE_CAUSE(aMsg, aTarget) \
|
||
|
A11YDEBUG_FOCUS_LOG_ACCTARGET("Caused by: " aMsg, aTarget)
|
||
|
|
||
|
#else
|
||
|
#define A11YDEBUG_FOCUS_LOG_DOMTARGET(aMsg, aTarget)
|
||
|
#define A11YDEBUG_FOCUS_LOG_ACCTARGET(aMsg, aTarget)
|
||
|
#define A11YDEBUG_FOCUS_LOG_WIDGET(aMsg, aWidget)
|
||
|
#define A11YDEBUG_FOCUS_NOTIFICATION_SUPPORTSTARGET(aMsg, aTargetMsg, aTarget)
|
||
|
#define A11YDEBUG_FOCUS_NOTIFICATION_DOMTARGET(aMsg, aTargetMsg, aTarget)
|
||
|
#define A11YDEBUG_FOCUS_NOTIFICATION_ACCTARGET(aMsg, aTargetMsg, aTarget)
|
||
|
#define A11YDEBUG_FOCUS_ACTIVEITEMCHANGE_CAUSE(aMsg, aTarget)
|
||
|
#endif
|
||
|
|
||
|
#endif
|