2009-02-18 23:06:14 -08:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim:expandtab:shiftwidth=2:tabstop=2:
|
|
|
|
*/
|
2012-05-21 04:12:37 -07:00
|
|
|
/* 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/. */
|
2009-02-18 23:06:14 -08:00
|
|
|
|
|
|
|
#include "nsTextEquivUtils.h"
|
|
|
|
|
2012-04-13 07:17:03 -07:00
|
|
|
#include "Accessible-inl.h"
|
2011-08-09 18:44:00 -07:00
|
|
|
#include "AccIterator.h"
|
2013-09-10 15:18:59 -07:00
|
|
|
#include "nsCoreUtils.h"
|
2009-02-18 23:06:14 -08:00
|
|
|
#include "nsIDOMXULLabeledControlEl.h"
|
|
|
|
|
2012-04-30 20:08:31 -07:00
|
|
|
using namespace mozilla::a11y;
|
|
|
|
|
2013-05-21 09:03:33 -07:00
|
|
|
/**
|
|
|
|
* The accessible for which we are computing a text equivalent. It is useful
|
|
|
|
* for bailing out during recursive text computation, or for special cases
|
|
|
|
* like step f. of the ARIA implementation guide.
|
|
|
|
*/
|
|
|
|
static Accessible* sInitiatorAcc = nullptr;
|
|
|
|
|
2009-02-18 23:06:14 -08:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsTextEquivUtils. Public.
|
|
|
|
|
|
|
|
nsresult
|
2012-05-28 18:18:45 -07:00
|
|
|
nsTextEquivUtils::GetNameFromSubtree(Accessible* aAccessible,
|
2009-02-18 23:06:14 -08:00
|
|
|
nsAString& aName)
|
|
|
|
{
|
|
|
|
aName.Truncate();
|
|
|
|
|
2013-05-21 09:03:33 -07:00
|
|
|
if (sInitiatorAcc)
|
2009-02-18 23:06:14 -08:00
|
|
|
return NS_OK;
|
|
|
|
|
2013-05-21 09:03:33 -07:00
|
|
|
sInitiatorAcc = aAccessible;
|
2012-11-15 02:58:05 -08:00
|
|
|
if (GetRoleRule(aAccessible->Role()) == eNameFromSubtreeRule) {
|
2010-06-11 01:23:18 -07:00
|
|
|
//XXX: is it necessary to care the accessible is not a document?
|
|
|
|
if (aAccessible->IsContent()) {
|
2009-02-18 23:06:14 -08:00
|
|
|
nsAutoString name;
|
|
|
|
AppendFromAccessibleChildren(aAccessible, &name);
|
|
|
|
name.CompressWhitespace();
|
2013-07-18 08:09:45 -07:00
|
|
|
if (!nsCoreUtils::IsWhitespaceString(name))
|
2009-06-12 18:06:02 -07:00
|
|
|
aName = name;
|
2009-02-18 23:06:14 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-21 09:03:33 -07:00
|
|
|
sInitiatorAcc = nullptr;
|
2009-02-18 23:06:14 -08:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
2012-05-28 18:18:45 -07:00
|
|
|
nsTextEquivUtils::GetTextEquivFromIDRefs(Accessible* aAccessible,
|
2009-02-18 23:06:14 -08:00
|
|
|
nsIAtom *aIDRefsAttr,
|
|
|
|
nsAString& aTextEquiv)
|
|
|
|
{
|
|
|
|
aTextEquiv.Truncate();
|
|
|
|
|
2010-06-11 01:23:18 -07:00
|
|
|
nsIContent* content = aAccessible->GetContent();
|
2009-02-18 23:06:14 -08:00
|
|
|
if (!content)
|
|
|
|
return NS_OK;
|
|
|
|
|
2012-07-30 07:20:58 -07:00
|
|
|
nsIContent* refContent = nullptr;
|
2012-03-22 22:26:52 -07:00
|
|
|
IDRefsIterator iter(aAccessible->Document(), content, aIDRefsAttr);
|
2010-11-16 19:32:15 -08:00
|
|
|
while ((refContent = iter.NextElem())) {
|
2009-02-18 23:06:14 -08:00
|
|
|
if (!aTextEquiv.IsEmpty())
|
|
|
|
aTextEquiv += ' ';
|
|
|
|
|
2010-11-16 19:32:15 -08:00
|
|
|
nsresult rv = AppendTextEquivFromContent(aAccessible, refContent,
|
|
|
|
&aTextEquiv);
|
2009-02-18 23:06:14 -08:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
2012-05-28 18:18:45 -07:00
|
|
|
nsTextEquivUtils::AppendTextEquivFromContent(Accessible* aInitiatorAcc,
|
2009-02-18 23:06:14 -08:00
|
|
|
nsIContent *aContent,
|
|
|
|
nsAString *aString)
|
|
|
|
{
|
|
|
|
// Prevent recursion which can cause infinite loops.
|
2013-05-21 09:03:33 -07:00
|
|
|
if (sInitiatorAcc)
|
2009-02-18 23:06:14 -08:00
|
|
|
return NS_OK;
|
|
|
|
|
2013-05-21 09:03:33 -07:00
|
|
|
sInitiatorAcc = aInitiatorAcc;
|
2009-02-18 23:06:14 -08:00
|
|
|
|
|
|
|
// If the given content is not visible or isn't accessible then go down
|
|
|
|
// through the DOM subtree otherwise go down through accessible subtree and
|
|
|
|
// calculate the flat string.
|
2009-12-24 13:20:05 -08:00
|
|
|
nsIFrame *frame = aContent->GetPrimaryFrame();
|
2013-02-16 13:51:02 -08:00
|
|
|
bool isVisible = frame && frame->StyleVisibility()->IsVisible();
|
2009-02-18 23:06:14 -08:00
|
|
|
|
2010-02-01 18:27:32 -08:00
|
|
|
nsresult rv = NS_ERROR_FAILURE;
|
2011-09-28 23:19:26 -07:00
|
|
|
bool goThroughDOMSubtree = true;
|
2009-02-18 23:06:14 -08:00
|
|
|
|
|
|
|
if (isVisible) {
|
2012-05-28 18:18:45 -07:00
|
|
|
Accessible* accessible =
|
2013-05-21 09:03:33 -07:00
|
|
|
sInitiatorAcc->Document()->GetAccessible(aContent);
|
2010-02-01 18:27:32 -08:00
|
|
|
if (accessible) {
|
2009-02-18 23:06:14 -08:00
|
|
|
rv = AppendFromAccessible(accessible, aString);
|
2011-10-17 07:59:28 -07:00
|
|
|
goThroughDOMSubtree = false;
|
2009-02-18 23:06:14 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (goThroughDOMSubtree)
|
|
|
|
rv = AppendFromDOMNode(aContent, aString);
|
|
|
|
|
2013-05-21 09:03:33 -07:00
|
|
|
sInitiatorAcc = nullptr;
|
2009-02-18 23:06:14 -08:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsTextEquivUtils::AppendTextEquivFromTextContent(nsIContent *aContent,
|
|
|
|
nsAString *aString)
|
|
|
|
{
|
|
|
|
if (aContent->IsNodeOfType(nsINode::eTEXT)) {
|
2011-09-28 23:19:26 -07:00
|
|
|
bool isHTMLBlock = false;
|
2010-06-11 01:23:18 -07:00
|
|
|
|
2013-05-01 15:50:08 -07:00
|
|
|
nsIContent *parentContent = aContent->GetFlattenedTreeParent();
|
2009-02-18 23:06:14 -08:00
|
|
|
if (parentContent) {
|
2009-12-24 13:20:05 -08:00
|
|
|
nsIFrame *frame = parentContent->GetPrimaryFrame();
|
2009-02-18 23:06:14 -08:00
|
|
|
if (frame) {
|
|
|
|
// If this text is inside a block level frame (as opposed to span
|
|
|
|
// level), we need to add spaces around that block's text, so we don't
|
|
|
|
// get words jammed together in final name.
|
2013-02-16 13:51:02 -08:00
|
|
|
const nsStyleDisplay* display = frame->StyleDisplay();
|
2012-08-02 04:38:51 -07:00
|
|
|
if (display->IsBlockOutsideStyle() ||
|
2009-02-18 23:06:14 -08:00
|
|
|
display->mDisplay == NS_STYLE_DISPLAY_TABLE_CELL) {
|
2011-10-17 07:59:28 -07:00
|
|
|
isHTMLBlock = true;
|
2009-02-18 23:06:14 -08:00
|
|
|
if (!aString->IsEmpty()) {
|
|
|
|
aString->Append(PRUnichar(' '));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aContent->TextLength() > 0) {
|
2009-12-24 13:20:05 -08:00
|
|
|
nsIFrame *frame = aContent->GetPrimaryFrame();
|
2009-02-18 23:06:14 -08:00
|
|
|
if (frame) {
|
|
|
|
nsresult rv = frame->GetRenderedText(aString);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
} else {
|
|
|
|
// If aContent is an object that is display: none, we have no a frame.
|
|
|
|
aContent->AppendTextTo(*aString);
|
|
|
|
}
|
|
|
|
if (isHTMLBlock && !aString->IsEmpty()) {
|
|
|
|
aString->Append(PRUnichar(' '));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2009-08-24 13:02:07 -07:00
|
|
|
if (aContent->IsHTML() &&
|
2011-06-03 14:35:17 -07:00
|
|
|
aContent->NodeInfo()->Equals(nsGkAtoms::br)) {
|
2009-02-18 23:06:14 -08:00
|
|
|
aString->AppendLiteral("\r\n");
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK_NO_NAME_CLAUSE_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsTextEquivUtils. Private.
|
|
|
|
|
|
|
|
nsresult
|
2012-05-28 18:18:45 -07:00
|
|
|
nsTextEquivUtils::AppendFromAccessibleChildren(Accessible* aAccessible,
|
2009-02-18 23:06:14 -08:00
|
|
|
nsAString *aString)
|
|
|
|
{
|
2009-02-27 02:54:39 -08:00
|
|
|
nsresult rv = NS_OK_NO_NAME_CLAUSE_HANDLED;
|
2009-02-18 23:06:14 -08:00
|
|
|
|
2012-08-22 08:56:38 -07:00
|
|
|
uint32_t childCount = aAccessible->ChildCount();
|
|
|
|
for (uint32_t childIdx = 0; childIdx < childCount; childIdx++) {
|
2012-05-28 18:18:45 -07:00
|
|
|
Accessible* child = aAccessible->GetChildAt(childIdx);
|
2010-05-18 07:03:56 -07:00
|
|
|
rv = AppendFromAccessible(child, aString);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
2009-02-18 23:06:14 -08:00
|
|
|
}
|
|
|
|
|
2009-02-27 02:54:39 -08:00
|
|
|
return rv;
|
2009-02-18 23:06:14 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
2012-05-28 18:18:45 -07:00
|
|
|
nsTextEquivUtils::AppendFromAccessible(Accessible* aAccessible,
|
2009-02-18 23:06:14 -08:00
|
|
|
nsAString *aString)
|
|
|
|
{
|
2010-06-11 01:23:18 -07:00
|
|
|
//XXX: is it necessary to care the accessible is not a document?
|
|
|
|
if (aAccessible->IsContent()) {
|
|
|
|
nsresult rv = AppendTextEquivFromTextContent(aAccessible->GetContent(),
|
|
|
|
aString);
|
2009-02-18 23:06:14 -08:00
|
|
|
if (rv != NS_OK_NO_NAME_CLAUSE_HANDLED)
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
bool isEmptyTextEquiv = true;
|
2009-02-27 02:54:39 -08:00
|
|
|
|
|
|
|
// If the name is from tooltip then append it to result string in the end
|
|
|
|
// (see h. step of name computation guide).
|
2012-04-30 20:08:31 -07:00
|
|
|
nsAutoString text;
|
|
|
|
if (aAccessible->Name(text) != eNameFromTooltip)
|
2009-02-27 02:54:39 -08:00
|
|
|
isEmptyTextEquiv = !AppendString(aString, text);
|
|
|
|
|
|
|
|
// Implementation of f. step.
|
2012-04-30 20:08:31 -07:00
|
|
|
nsresult rv = AppendFromValue(aAccessible, aString);
|
2009-02-27 02:54:39 -08:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
if (rv != NS_OK_NO_NAME_CLAUSE_HANDLED)
|
2011-10-17 07:59:28 -07:00
|
|
|
isEmptyTextEquiv = false;
|
2009-02-27 02:54:39 -08:00
|
|
|
|
|
|
|
// Implementation of g) step of text equivalent computation guide. Go down
|
|
|
|
// into subtree if accessible allows "text equivalent from subtree rule" or
|
|
|
|
// it's not root and not control.
|
|
|
|
if (isEmptyTextEquiv) {
|
2012-08-22 08:56:38 -07:00
|
|
|
uint32_t nameRule = GetRoleRule(aAccessible->Role());
|
2012-11-15 02:58:05 -08:00
|
|
|
if (nameRule & eNameFromSubtreeIfReqRule) {
|
2009-02-27 02:54:39 -08:00
|
|
|
rv = AppendFromAccessibleChildren(aAccessible, aString);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
if (rv != NS_OK_NO_NAME_CLAUSE_HANDLED)
|
2011-10-17 07:59:28 -07:00
|
|
|
isEmptyTextEquiv = false;
|
2009-02-27 02:54:39 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Implementation of h. step
|
|
|
|
if (isEmptyTextEquiv && !text.IsEmpty()) {
|
|
|
|
AppendString(aString, text);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
2012-05-28 18:18:45 -07:00
|
|
|
nsTextEquivUtils::AppendFromValue(Accessible* aAccessible,
|
2009-02-27 02:54:39 -08:00
|
|
|
nsAString *aString)
|
|
|
|
{
|
2012-11-15 02:58:05 -08:00
|
|
|
if (GetRoleRule(aAccessible->Role()) != eNameFromValueRule)
|
2009-02-27 02:54:39 -08:00
|
|
|
return NS_OK_NO_NAME_CLAUSE_HANDLED;
|
2009-02-18 23:06:14 -08:00
|
|
|
|
2009-02-27 02:54:39 -08:00
|
|
|
// Implementation of step f. of text equivalent computation. If the given
|
|
|
|
// accessible is not root accessible (the accessible the text equivalent is
|
|
|
|
// computed for in the end) then append accessible value. Otherwise append
|
|
|
|
// value if and only if the given accessible is in the middle of its parent.
|
2009-02-18 23:06:14 -08:00
|
|
|
|
2009-02-27 02:54:39 -08:00
|
|
|
nsAutoString text;
|
2013-05-21 09:03:33 -07:00
|
|
|
if (aAccessible != sInitiatorAcc) {
|
2012-04-09 02:48:41 -07:00
|
|
|
aAccessible->Value(text);
|
2009-02-27 02:54:39 -08:00
|
|
|
|
|
|
|
return AppendString(aString, text) ?
|
|
|
|
NS_OK : NS_OK_NO_NAME_CLAUSE_HANDLED;
|
|
|
|
}
|
|
|
|
|
2010-06-11 01:23:18 -07:00
|
|
|
//XXX: is it necessary to care the accessible is not a document?
|
2013-03-13 22:33:37 -07:00
|
|
|
if (aAccessible->IsDoc())
|
2010-06-11 01:23:18 -07:00
|
|
|
return NS_ERROR_UNEXPECTED;
|
2009-07-29 02:03:20 -07:00
|
|
|
|
2010-06-11 01:23:18 -07:00
|
|
|
nsIContent *content = aAccessible->GetContent();
|
2009-07-29 02:03:20 -07:00
|
|
|
|
2011-12-06 23:20:17 -08:00
|
|
|
for (nsIContent* childContent = content->GetPreviousSibling(); childContent;
|
|
|
|
childContent = childContent->GetPreviousSibling()) {
|
2009-07-29 02:03:20 -07:00
|
|
|
// check for preceding text...
|
2011-12-06 23:20:17 -08:00
|
|
|
if (!childContent->TextIsOnlyWhitespace()) {
|
|
|
|
for (nsIContent* siblingContent = content->GetNextSibling(); siblingContent;
|
|
|
|
siblingContent = siblingContent->GetNextSibling()) {
|
2009-07-29 02:03:20 -07:00
|
|
|
// .. and subsequent text
|
2011-12-06 23:20:17 -08:00
|
|
|
if (!siblingContent->TextIsOnlyWhitespace()) {
|
2012-04-09 02:48:41 -07:00
|
|
|
aAccessible->Value(text);
|
2009-07-29 02:03:20 -07:00
|
|
|
|
|
|
|
return AppendString(aString, text) ?
|
|
|
|
NS_OK : NS_OK_NO_NAME_CLAUSE_HANDLED;
|
|
|
|
break;
|
|
|
|
}
|
2009-02-18 23:06:14 -08:00
|
|
|
}
|
2009-07-29 02:03:20 -07:00
|
|
|
break;
|
2009-02-18 23:06:14 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-27 02:54:39 -08:00
|
|
|
return NS_OK_NO_NAME_CLAUSE_HANDLED;
|
2009-02-18 23:06:14 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsTextEquivUtils::AppendFromDOMChildren(nsIContent *aContent,
|
|
|
|
nsAString *aString)
|
|
|
|
{
|
2011-12-06 23:20:17 -08:00
|
|
|
for (nsIContent* childContent = aContent->GetFirstChild(); childContent;
|
|
|
|
childContent = childContent->GetNextSibling()) {
|
2009-02-18 23:06:14 -08:00
|
|
|
nsresult rv = AppendFromDOMNode(childContent, aString);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsTextEquivUtils::AppendFromDOMNode(nsIContent *aContent, nsAString *aString)
|
|
|
|
{
|
|
|
|
nsresult rv = AppendTextEquivFromTextContent(aContent, aString);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
if (rv != NS_OK_NO_NAME_CLAUSE_HANDLED)
|
|
|
|
return NS_OK;
|
|
|
|
|
2009-08-24 13:02:07 -07:00
|
|
|
if (aContent->IsXUL()) {
|
2009-02-18 23:06:14 -08:00
|
|
|
nsAutoString textEquivalent;
|
|
|
|
nsCOMPtr<nsIDOMXULLabeledControlElement> labeledEl =
|
|
|
|
do_QueryInterface(aContent);
|
|
|
|
|
|
|
|
if (labeledEl) {
|
|
|
|
labeledEl->GetLabel(textEquivalent);
|
|
|
|
} else {
|
2011-06-03 14:35:17 -07:00
|
|
|
if (aContent->NodeInfo()->Equals(nsGkAtoms::label,
|
2009-02-18 23:06:14 -08:00
|
|
|
kNameSpaceID_XUL))
|
2011-06-03 14:35:17 -07:00
|
|
|
aContent->GetAttr(kNameSpaceID_None, nsGkAtoms::value,
|
2009-02-18 23:06:14 -08:00
|
|
|
textEquivalent);
|
|
|
|
|
|
|
|
if (textEquivalent.IsEmpty())
|
|
|
|
aContent->GetAttr(kNameSpaceID_None,
|
2011-06-03 14:35:17 -07:00
|
|
|
nsGkAtoms::tooltiptext, textEquivalent);
|
2009-02-18 23:06:14 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
AppendString(aString, textEquivalent);
|
|
|
|
}
|
|
|
|
|
|
|
|
return AppendFromDOMChildren(aContent, aString);
|
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
bool
|
2009-02-18 23:06:14 -08:00
|
|
|
nsTextEquivUtils::AppendString(nsAString *aString,
|
|
|
|
const nsAString& aTextEquivalent)
|
|
|
|
{
|
|
|
|
if (aTextEquivalent.IsEmpty())
|
2011-10-17 07:59:28 -07:00
|
|
|
return false;
|
2009-02-18 23:06:14 -08:00
|
|
|
|
2013-02-12 01:13:57 -08:00
|
|
|
// Insert spaces to insure that words from controls aren't jammed together.
|
2013-07-18 08:09:45 -07:00
|
|
|
if (!aString->IsEmpty() && !nsCoreUtils::IsWhitespace(aString->Last()))
|
2009-02-18 23:06:14 -08:00
|
|
|
aString->Append(PRUnichar(' '));
|
|
|
|
|
|
|
|
aString->Append(aTextEquivalent);
|
2013-02-12 01:13:57 -08:00
|
|
|
|
2013-07-18 08:09:45 -07:00
|
|
|
if (!nsCoreUtils::IsWhitespace(aString->Last()))
|
2013-02-12 01:13:57 -08:00
|
|
|
aString->Append(PRUnichar(' '));
|
|
|
|
|
2011-10-17 07:59:28 -07:00
|
|
|
return true;
|
2009-02-18 23:06:14 -08:00
|
|
|
}
|
|
|
|
|
2012-08-22 08:56:38 -07:00
|
|
|
uint32_t
|
2012-07-21 11:32:25 -07:00
|
|
|
nsTextEquivUtils::GetRoleRule(role aRole)
|
2009-02-18 23:06:14 -08:00
|
|
|
{
|
2012-07-21 11:32:25 -07:00
|
|
|
#define ROLE(geckoRole, stringRole, atkRole, \
|
|
|
|
macRole, msaaRole, ia2Role, nameRule) \
|
|
|
|
case roles::geckoRole: \
|
|
|
|
return nameRule;
|
|
|
|
|
|
|
|
switch (aRole) {
|
|
|
|
#include "RoleMap.h"
|
|
|
|
default:
|
2013-06-28 18:38:30 -07:00
|
|
|
MOZ_CRASH("Unknown role.");
|
2012-07-21 11:32:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#undef ROLE
|
|
|
|
}
|
|
|
|
|