2007-06-11 23:10:23 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 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/. */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
#ifndef nsMathMLFrame_h___
|
|
|
|
#define nsMathMLFrame_h___
|
|
|
|
|
2012-09-14 09:10:08 -07:00
|
|
|
#include "mozilla/Attributes.h"
|
2011-04-07 18:05:49 -07:00
|
|
|
#include "nsFontMetrics.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "nsMathMLOperators.h"
|
|
|
|
#include "nsIMathMLFrame.h"
|
2011-08-14 10:08:04 -07:00
|
|
|
#include "nsLayoutUtils.h"
|
2013-09-30 14:26:04 -07:00
|
|
|
#include "nsBoundingMetrics.h"
|
|
|
|
#include "nsIFrame.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
class nsMathMLChar;
|
2013-09-30 14:26:04 -07:00
|
|
|
class nsCSSValue;
|
|
|
|
class nsDisplayListSet;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// Concrete base class with default methods that derived MathML frames can override
|
|
|
|
class nsMathMLFrame : public nsIMathMLFrame {
|
|
|
|
public:
|
|
|
|
|
|
|
|
// nsIMathMLFrame ---
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
virtual bool
|
2013-05-14 09:33:23 -07:00
|
|
|
IsSpaceLike() MOZ_OVERRIDE {
|
2011-03-24 20:38:59 -07:00
|
|
|
return NS_MATHML_IS_SPACE_LIKE(mPresentationData.flags);
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
NS_IMETHOD
|
2012-09-14 09:10:08 -07:00
|
|
|
GetBoundingMetrics(nsBoundingMetrics& aBoundingMetrics) MOZ_OVERRIDE {
|
2007-03-22 10:30:00 -07:00
|
|
|
aBoundingMetrics = mBoundingMetrics;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHOD
|
2012-09-14 09:10:08 -07:00
|
|
|
SetBoundingMetrics(const nsBoundingMetrics& aBoundingMetrics) MOZ_OVERRIDE {
|
2007-03-22 10:30:00 -07:00
|
|
|
mBoundingMetrics = aBoundingMetrics;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHOD
|
2012-09-14 09:10:08 -07:00
|
|
|
SetReference(const nsPoint& aReference) MOZ_OVERRIDE {
|
2007-03-22 10:30:00 -07:00
|
|
|
mReference = aReference;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2012-09-14 09:10:08 -07:00
|
|
|
virtual eMathMLFrameType GetMathMLFrameType() MOZ_OVERRIDE;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
NS_IMETHOD
|
2011-04-07 18:04:40 -07:00
|
|
|
Stretch(nsRenderingContext& aRenderingContext,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsStretchDirection aStretchDirection,
|
|
|
|
nsBoundingMetrics& aContainerSize,
|
2012-09-14 09:10:08 -07:00
|
|
|
nsHTMLReflowMetrics& aDesiredStretchSize) MOZ_OVERRIDE
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHOD
|
2012-09-14 09:10:08 -07:00
|
|
|
GetEmbellishData(nsEmbellishData& aEmbellishData) MOZ_OVERRIDE {
|
2007-03-22 10:30:00 -07:00
|
|
|
aEmbellishData = mEmbellishData;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHOD
|
2012-09-14 09:10:08 -07:00
|
|
|
GetPresentationData(nsPresentationData& aPresentationData) MOZ_OVERRIDE {
|
2007-03-22 10:30:00 -07:00
|
|
|
aPresentationData = mPresentationData;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHOD
|
2012-09-14 09:10:08 -07:00
|
|
|
InheritAutomaticData(nsIFrame* aParent) MOZ_OVERRIDE;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
NS_IMETHOD
|
2012-09-14 09:10:08 -07:00
|
|
|
TransmitAutomaticData() MOZ_OVERRIDE
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHOD
|
2012-08-22 08:56:38 -07:00
|
|
|
UpdatePresentationData(uint32_t aFlagsValues,
|
2012-09-14 09:10:08 -07:00
|
|
|
uint32_t aFlagsToUpdate) MOZ_OVERRIDE;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
NS_IMETHOD
|
2012-08-22 08:56:38 -07:00
|
|
|
UpdatePresentationDataFromChildAt(int32_t aFirstIndex,
|
|
|
|
int32_t aLastIndex,
|
|
|
|
uint32_t aFlagsValues,
|
2012-09-14 09:10:08 -07:00
|
|
|
uint32_t aFlagsToUpdate) MOZ_OVERRIDE
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// helper to give a style context suitable for doing the stretching to the
|
|
|
|
// MathMLChar. Frame classes that use this should make the extra style contexts
|
|
|
|
// accessible to the Style System via Get/Set AdditionalStyleContext.
|
|
|
|
static void
|
|
|
|
ResolveMathMLCharStyle(nsPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsStyleContext* aParenStyleContext,
|
|
|
|
nsMathMLChar* aMathMLChar,
|
2011-09-28 23:19:26 -07:00
|
|
|
bool aIsMutableChar);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// helper to get the mEmbellishData of a frame
|
|
|
|
// The MathML REC precisely defines an "embellished operator" as:
|
|
|
|
// - an <mo> element;
|
|
|
|
// - or one of the elements <msub>, <msup>, <msubsup>, <munder>, <mover>,
|
|
|
|
// <munderover>, <mmultiscripts>, <mfrac>, or <semantics>, whose first
|
|
|
|
// argument exists and is an embellished operator;
|
|
|
|
//- or one of the elements <mstyle>, <mphantom>, or <mpadded>, such that
|
|
|
|
// an <mrow> containing the same arguments would be an embellished
|
|
|
|
// operator;
|
|
|
|
// - or an <maction> element whose selected subexpression exists and is an
|
|
|
|
// embellished operator;
|
|
|
|
// - or an <mrow> whose arguments consist (in any order) of one embellished
|
|
|
|
// operator and zero or more spacelike elements.
|
|
|
|
static void
|
|
|
|
GetEmbellishDataFrom(nsIFrame* aFrame,
|
|
|
|
nsEmbellishData& aEmbellishData);
|
|
|
|
|
|
|
|
// helper to get the presentation data of a frame. If aClimbTree is
|
|
|
|
// set to true and the frame happens to be surrounded by non-MathML
|
|
|
|
// helper frames needed for its support, we walk up the frame hierarchy
|
|
|
|
// until we reach a MathML ancestor or the <root> math element.
|
|
|
|
static void
|
|
|
|
GetPresentationDataFrom(nsIFrame* aFrame,
|
|
|
|
nsPresentationData& aPresentationData,
|
2011-09-28 23:19:26 -07:00
|
|
|
bool aClimbTree = true);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// helper used by <mstyle> and <mtable> to see if they have a displaystyle attribute
|
|
|
|
static void
|
|
|
|
FindAttrDisplaystyle(nsIContent* aContent,
|
|
|
|
nsPresentationData& aPresentationData);
|
|
|
|
|
2012-07-30 07:20:58 -07:00
|
|
|
// helper to check if a content has an attribute. If content is nullptr or if
|
2007-03-22 10:30:00 -07:00
|
|
|
// the attribute is not there, check if the attribute is on the mstyle hierarchy
|
2011-10-17 07:59:28 -07:00
|
|
|
// @return true --if attribute exists
|
|
|
|
// false --if attribute doesn't exist
|
2011-09-28 23:19:26 -07:00
|
|
|
static bool
|
2007-03-22 10:30:00 -07:00
|
|
|
GetAttribute(nsIContent* aContent,
|
|
|
|
nsIFrame* aMathMLmstyleFrame,
|
|
|
|
nsIAtom* aAttributeAtom,
|
|
|
|
nsString& aValue);
|
|
|
|
|
|
|
|
// utilities to parse and retrieve numeric values in CSS units
|
|
|
|
// All values are stored in twips.
|
2012-05-15 15:30:14 -07:00
|
|
|
// @pre aLengthValue is the default length value of the attribute.
|
|
|
|
// @post aLengthValue is the length value computed from the attribute.
|
|
|
|
static void ParseNumericValue(const nsString& aString,
|
|
|
|
nscoord* aLengthValue,
|
2012-08-22 08:56:38 -07:00
|
|
|
uint32_t aFlags,
|
2012-05-15 15:30:14 -07:00
|
|
|
nsPresContext* aPresContext,
|
|
|
|
nsStyleContext* aStyleContext);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
static nscoord
|
|
|
|
CalcLength(nsPresContext* aPresContext,
|
|
|
|
nsStyleContext* aStyleContext,
|
|
|
|
const nsCSSValue& aCSSValue);
|
|
|
|
|
|
|
|
static eMathMLFrameType
|
|
|
|
GetMathMLFrameTypeFor(nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
if (aFrame->IsFrameOfType(nsIFrame::eMathML)) {
|
2009-01-12 11:20:59 -08:00
|
|
|
nsIMathMLFrame* mathMLFrame = do_QueryFrame(aFrame);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (mathMLFrame)
|
|
|
|
return mathMLFrame->GetMathMLFrameType();
|
|
|
|
}
|
|
|
|
return eMathMLFrameType_UNKNOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
// estimate of the italic correction
|
|
|
|
static void
|
|
|
|
GetItalicCorrection(nsBoundingMetrics& aBoundingMetrics,
|
|
|
|
nscoord& aItalicCorrection)
|
|
|
|
{
|
|
|
|
aItalicCorrection = aBoundingMetrics.rightBearing - aBoundingMetrics.width;
|
|
|
|
if (0 > aItalicCorrection) {
|
|
|
|
aItalicCorrection = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
GetItalicCorrection(nsBoundingMetrics& aBoundingMetrics,
|
|
|
|
nscoord& aLeftItalicCorrection,
|
|
|
|
nscoord& aRightItalicCorrection)
|
|
|
|
{
|
|
|
|
aRightItalicCorrection = aBoundingMetrics.rightBearing - aBoundingMetrics.width;
|
|
|
|
if (0 > aRightItalicCorrection) {
|
|
|
|
aRightItalicCorrection = 0;
|
|
|
|
}
|
|
|
|
aLeftItalicCorrection = -aBoundingMetrics.leftBearing;
|
|
|
|
if (0 > aLeftItalicCorrection) {
|
|
|
|
aLeftItalicCorrection = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// helper methods for getting sup/subdrop's from a child
|
|
|
|
static void
|
|
|
|
GetSubDropFromChild(nsIFrame* aChild,
|
|
|
|
nscoord& aSubDrop)
|
|
|
|
{
|
2011-08-14 10:08:04 -07:00
|
|
|
nsRefPtr<nsFontMetrics> fm;
|
|
|
|
nsLayoutUtils::GetFontMetricsForFrame(aChild, getter_AddRefs(fm));
|
2007-03-22 10:30:00 -07:00
|
|
|
GetSubDrop(fm, aSubDrop);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
GetSupDropFromChild(nsIFrame* aChild,
|
|
|
|
nscoord& aSupDrop)
|
|
|
|
{
|
2011-08-14 10:08:04 -07:00
|
|
|
nsRefPtr<nsFontMetrics> fm;
|
|
|
|
nsLayoutUtils::GetFontMetricsForFrame(aChild, getter_AddRefs(fm));
|
2007-03-22 10:30:00 -07:00
|
|
|
GetSupDrop(fm, aSupDrop);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
GetSkewCorrectionFromChild(nsIFrame* aChild,
|
|
|
|
nscoord& aSkewCorrection)
|
|
|
|
{
|
|
|
|
// default is 0
|
|
|
|
// individual classes should over-ride this method if necessary
|
|
|
|
aSkewCorrection = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 2 levels of subscript shifts
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetSubScriptShifts(nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& aSubScriptShift1,
|
|
|
|
nscoord& aSubScriptShift2)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
nscoord xHeight = fm->XHeight();
|
2007-03-22 10:30:00 -07:00
|
|
|
aSubScriptShift1 = NSToCoordRound(150.000f/430.556f * xHeight);
|
|
|
|
aSubScriptShift2 = NSToCoordRound(247.217f/430.556f * xHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
// 3 levels of superscript shifts
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetSupScriptShifts(nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& aSupScriptShift1,
|
|
|
|
nscoord& aSupScriptShift2,
|
|
|
|
nscoord& aSupScriptShift3)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
nscoord xHeight = fm->XHeight();
|
2007-03-22 10:30:00 -07:00
|
|
|
aSupScriptShift1 = NSToCoordRound(412.892f/430.556f * xHeight);
|
|
|
|
aSupScriptShift2 = NSToCoordRound(362.892f/430.556f * xHeight);
|
|
|
|
aSupScriptShift3 = NSToCoordRound(288.889f/430.556f * xHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
// these are TeX specific params not found in ordinary fonts
|
|
|
|
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetSubDrop(nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& aSubDrop)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
nscoord xHeight = fm->XHeight();
|
2007-03-22 10:30:00 -07:00
|
|
|
aSubDrop = NSToCoordRound(50.000f/430.556f * xHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetSupDrop(nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& aSupDrop)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
nscoord xHeight = fm->XHeight();
|
2007-03-22 10:30:00 -07:00
|
|
|
aSupDrop = NSToCoordRound(386.108f/430.556f * xHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetNumeratorShifts(nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& numShift1,
|
|
|
|
nscoord& numShift2,
|
|
|
|
nscoord& numShift3)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
nscoord xHeight = fm->XHeight();
|
2007-03-22 10:30:00 -07:00
|
|
|
numShift1 = NSToCoordRound(676.508f/430.556f * xHeight);
|
|
|
|
numShift2 = NSToCoordRound(393.732f/430.556f * xHeight);
|
|
|
|
numShift3 = NSToCoordRound(443.731f/430.556f * xHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetDenominatorShifts(nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& denShift1,
|
|
|
|
nscoord& denShift2)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
nscoord xHeight = fm->XHeight();
|
2007-03-22 10:30:00 -07:00
|
|
|
denShift1 = NSToCoordRound(685.951f/430.556f * xHeight);
|
|
|
|
denShift2 = NSToCoordRound(344.841f/430.556f * xHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetEmHeight(nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& emHeight)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
#if 0
|
2007-03-22 10:30:00 -07:00
|
|
|
// should switch to this API in order to scale with changes of TextZoom
|
2011-04-07 21:18:43 -07:00
|
|
|
emHeight = fm->EmHeight();
|
2007-03-22 10:30:00 -07:00
|
|
|
#else
|
|
|
|
emHeight = NSToCoordRound(float(fm->Font().size));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetAxisHeight (nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& axisHeight)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
axisHeight = NSToCoordRound(250.000f/430.556f * fm->XHeight());
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetBigOpSpacings(nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& bigOpSpacing1,
|
|
|
|
nscoord& bigOpSpacing2,
|
|
|
|
nscoord& bigOpSpacing3,
|
|
|
|
nscoord& bigOpSpacing4,
|
|
|
|
nscoord& bigOpSpacing5)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
nscoord xHeight = fm->XHeight();
|
2007-03-22 10:30:00 -07:00
|
|
|
bigOpSpacing1 = NSToCoordRound(111.111f/430.556f * xHeight);
|
|
|
|
bigOpSpacing2 = NSToCoordRound(166.667f/430.556f * xHeight);
|
|
|
|
bigOpSpacing3 = NSToCoordRound(200.000f/430.556f * xHeight);
|
|
|
|
bigOpSpacing4 = NSToCoordRound(600.000f/430.556f * xHeight);
|
|
|
|
bigOpSpacing5 = NSToCoordRound(100.000f/430.556f * xHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-04-07 18:05:49 -07:00
|
|
|
GetRuleThickness(nsFontMetrics* fm,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& ruleThickness)
|
|
|
|
{
|
2011-04-07 21:18:43 -07:00
|
|
|
nscoord xHeight = fm->XHeight();
|
2007-03-22 10:30:00 -07:00
|
|
|
ruleThickness = NSToCoordRound(40.000f/430.556f * xHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Some parameters are not accurately obtained using the x-height.
|
|
|
|
// Here are some slower variants to obtain the desired metrics
|
|
|
|
// by actually measuring some characters
|
|
|
|
static void
|
2011-04-07 18:04:40 -07:00
|
|
|
GetRuleThickness(nsRenderingContext& aRenderingContext,
|
2011-04-07 18:05:49 -07:00
|
|
|
nsFontMetrics* aFontMetrics,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& aRuleThickness);
|
|
|
|
|
|
|
|
static void
|
2011-04-07 18:04:40 -07:00
|
|
|
GetAxisHeight(nsRenderingContext& aRenderingContext,
|
2011-04-07 18:05:49 -07:00
|
|
|
nsFontMetrics* aFontMetrics,
|
2007-03-22 10:30:00 -07:00
|
|
|
nscoord& aAxisHeight);
|
|
|
|
|
|
|
|
protected:
|
2012-06-25 12:59:42 -07:00
|
|
|
#if defined(DEBUG) && defined(SHOW_BOUNDING_BOX)
|
2007-03-22 10:30:00 -07:00
|
|
|
nsresult DisplayBoundingMetrics(nsDisplayListBuilder* aBuilder,
|
|
|
|
nsIFrame* aFrame, const nsPoint& aPt,
|
|
|
|
const nsBoundingMetrics& aMetrics,
|
|
|
|
const nsDisplayListSet& aLists);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Display a solid rectangle in the frame's text color. Used for drawing
|
|
|
|
* fraction separators and root/sqrt overbars.
|
|
|
|
*/
|
2013-02-14 03:12:27 -08:00
|
|
|
void DisplayBar(nsDisplayListBuilder* aBuilder,
|
|
|
|
nsIFrame* aFrame, const nsRect& aRect,
|
|
|
|
const nsDisplayListSet& aLists);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// information about the presentation policy of the frame
|
|
|
|
nsPresentationData mPresentationData;
|
|
|
|
|
|
|
|
// information about a container that is an embellished operator
|
|
|
|
nsEmbellishData mEmbellishData;
|
|
|
|
|
|
|
|
// Metrics that _exactly_ enclose the text of the frame
|
|
|
|
nsBoundingMetrics mBoundingMetrics;
|
|
|
|
|
|
|
|
// Reference point of the frame: mReference.y is the baseline
|
|
|
|
nsPoint mReference;
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif /* nsMathMLFrame_h___ */
|