/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * 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/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0 . */ #ifndef INCLUDED_STARMATH_INC_NODE_HXX #define INCLUDED_STARMATH_INC_NODE_HXX #include "types.hxx" #include "token.hxx" #include "error.hxx" #include "rect.hxx" #include "format.hxx" #include #include #include #include #define ATTR_BOLD 0x0001 #define ATTR_ITALIC 0x0002 enum class FontSizeType { ABSOLUT = 1, PLUS = 2, MINUS = 3, MULTIPLY = 4, DIVIDE = 5 }; // flags to interdict respective status changes #define FLG_FONT 0x0001 #define FLG_SIZE 0x0002 #define FLG_BOLD 0x0004 #define FLG_ITALIC 0x0008 #define FLG_COLOR 0x0010 #define FLG_VISIBLE 0x0020 #define FLG_HORALIGN 0x0040 class SmVisitor; class SmDocShell; class SmNode; class SmStructureNode; typedef std::shared_ptr SmNodePointer; typedef std::deque> SmNodeStack; typedef std::vector< SmNode * > SmNodeArray; template < typename T > T* popOrZero(std::deque> & rStack) { if (rStack.empty()) return nullptr; std::unique_ptr pTmp(std::move(rStack.front())); rStack.pop_front(); return pTmp.release(); } enum SmScaleMode { SCALE_NONE, SCALE_WIDTH, SCALE_HEIGHT }; enum SmNodeType { /* 0*/ NTABLE, NBRACE, NBRACEBODY, NOPER, NALIGN, /* 5*/ NATTRIBUT, NFONT, NUNHOR, NBINHOR, NBINVER, /*10*/ NBINDIAGONAL, NSUBSUP, NMATRIX, NPLACE, NTEXT, /*15*/ NSPECIAL, NGLYPH_SPECIAL, NMATH, NBLANK, NERROR, /*20*/ NLINE, NEXPRESSION, NPOLYLINE, NROOT, NROOTSYMBOL, /*25*/ NRECTANGLE, NVERTICAL_BRACE, NMATHIDENT, NDYNINT, NDYNINTSYMBOL }; class SmNode : public SmRect { SmFace maFace; SmToken maNodeToken; SmNodeType meType; SmScaleMode meScaleMode; RectHorAlign meRectHorAlign; sal_uInt16 mnFlags, mnAttributes; bool mbIsPhantom; bool mbIsSelected; protected: SmNode(SmNodeType eNodeType, const SmToken &rNodeToken); // index in accessible text -1 if not (yet) applicable sal_Int32 mnAccIndex; public: SmNode(const SmNode&) = delete; SmNode& operator=(const SmNode&) = delete; virtual ~SmNode(); virtual bool IsVisible() const; virtual sal_uInt16 GetNumSubNodes() const; virtual SmNode * GetSubNode(sal_uInt16 nIndex); const SmNode * GetSubNode(sal_uInt16 nIndex) const { return const_cast(this)->GetSubNode(nIndex); } virtual SmNode * GetLeftMost(); const SmNode * GetLeftMost() const { return const_cast(this)->GetLeftMost(); } sal_uInt16 & Flags() { return mnFlags; } sal_uInt16 & Attributes() { return mnAttributes; } bool IsPhantom() const { return mbIsPhantom; } void SetPhantom(bool bIsPhantom); void SetColor(const Color &rColor); void SetAttribut(sal_uInt16 nAttrib); void ClearAttribut(sal_uInt16 nAttrib); const SmFace & GetFont() const { return maFace; }; SmFace & GetFont() { return maFace; }; void SetFont(const SmFace &rFace); void SetFontSize(const Fraction &rRelSize, FontSizeType nType); void SetSize(const Fraction &rScale); virtual void Prepare(const SmFormat &rFormat, const SmDocShell &rDocShell); void PrepareAttributes(); void SetRectHorAlign(RectHorAlign eHorAlign, bool bApplyToSubTree = true ); RectHorAlign GetRectHorAlign() const { return meRectHorAlign; } const SmRect & GetRect() const { return *this; } SmRect & GetRect() { return *this; } void Move(const Point &rPosition); void MoveTo(const Point &rPosition) { Move(rPosition - GetTopLeft()); } virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat); virtual void CreateTextFromNode(OUString &rText); virtual void GetAccessibleText( OUStringBuffer &rText ) const = 0; sal_Int32 GetAccessibleIndex() const { return mnAccIndex; } const SmNode * FindNodeWithAccessibleIndex(sal_Int32 nAccIndex) const; sal_uInt16 GetRow() const { return sal::static_int_cast(maNodeToken.nRow); } sal_uInt16 GetColumn() const { return sal::static_int_cast(maNodeToken.nCol); } SmScaleMode GetScaleMode() const { return meScaleMode; } void SetScaleMode(SmScaleMode eMode) { meScaleMode = eMode; } virtual void AdaptToX(OutputDevice &rDev, sal_uLong nWidth); virtual void AdaptToY(OutputDevice &rDev, sal_uLong nHeight); SmNodeType GetType() const { return meType; } const SmToken & GetToken() const { return maNodeToken; } const SmNode * FindTokenAt(sal_uInt16 nRow, sal_uInt16 nCol) const; const SmNode * FindRectClosestTo(const Point &rPoint) const; virtual long GetFormulaBaseline() const; /** Accept a visitor * Calls the method for this class on the visitor */ virtual void Accept(SmVisitor* pVisitor) = 0; /** True if this node is selected */ bool IsSelected() const {return mbIsSelected;} void SetSelected(bool Selected = true) {mbIsSelected = Selected;} /** Get the parent node of this node */ SmStructureNode* GetParent(){ return mpParentNode; } const SmStructureNode* GetParent() const { return mpParentNode; } /** Set the parent node */ void SetParent(SmStructureNode* parent){ mpParentNode = parent; } /** Get the index of a child node * * Returns -1, if pSubNode isn't a subnode of this. */ int IndexOfSubNode(SmNode* pSubNode){ sal_uInt16 nSize = GetNumSubNodes(); for(sal_uInt16 i = 0; i < nSize; i++) if(pSubNode == GetSubNode(i)) return i; return -1; } /** Set the token for this node */ void SetToken(SmToken& token){ maNodeToken = token; } private: SmStructureNode* mpParentNode; }; /** A simple auxiliary iterator class for SmNode * * Example of iteration over children of pMyNode: * \code * //Node to iterate over: * SmNode* pMyNode = 0;// A pointer from somewhere * //The iterator: * SmNodeIterator it(pMyNode); * //The iteration: * while(it.Next()) { * it->SetSelected(true); * } * \endcode */ class SmNodeIterator{ public: SmNodeIterator(SmNode* node, bool bReverse = false){ pNode = node; nSize = pNode->GetNumSubNodes(); nIndex = 0; pChildNode = nullptr; bIsReverse = bReverse; } /** Get the subnode or NULL if none */ SmNode* Next(){ while(!bIsReverse && nIndex < nSize){ if(nullptr != (pChildNode = pNode->GetSubNode(nIndex++))) return pChildNode; } while(bIsReverse && nSize > 0){ if(nullptr != (pChildNode = pNode->GetSubNode((nSize--)-1))) return pChildNode; } pChildNode = nullptr; return nullptr; } /** Get the current child node, NULL if none */ SmNode* Current(){ return pChildNode; } /** Get the current child node, NULL if none */ SmNode* operator->(){ return pChildNode; } private: /** Current child */ SmNode* pChildNode; /** Node whos children we're iterating over */ SmNode* pNode; /** Size of the node */ sal_uInt16 nSize; /** Current index in the node */ sal_uInt16 nIndex; /** Move reverse */ bool bIsReverse; }; /** Abstract baseclass for all composite node * * Subclasses of this class can have subnodes. Nodes that doesn't derivate from * this class does not have subnodes. */ class SmStructureNode : public SmNode { SmNodeArray aSubNodes; protected: SmStructureNode(SmNodeType eNodeType, const SmToken &rNodeToken) : SmNode(eNodeType, rNodeToken) {} public: virtual ~SmStructureNode(); virtual bool IsVisible() const override; virtual sal_uInt16 GetNumSubNodes() const override; void SetNumSubNodes(sal_uInt16 nSize) { aSubNodes.resize(nSize); } using SmNode::GetSubNode; virtual SmNode * GetSubNode(sal_uInt16 nIndex) override; void SetSubNodes(SmNode *pFirst, SmNode *pSecond, SmNode *pThird = nullptr); void SetSubNodes(const SmNodeArray &rNodeArray); virtual void GetAccessibleText( OUStringBuffer &rText ) const override; void SetSubNode(size_t nIndex, SmNode* pNode) { size_t size = aSubNodes.size(); if (size <= nIndex) { //Resize subnodes array aSubNodes.resize(nIndex + 1); //Set new slots to NULL for (size_t i = size; i < nIndex+1; i++) aSubNodes[i] = nullptr; } aSubNodes[nIndex] = pNode; ClaimPaternity(); } private: /** Sets parent on children of this node */ void ClaimPaternity(); }; /** Abstract base class for all visible node * * Nodes that doesn't derivate from this class doesn't draw anything, but their * children. */ class SmVisibleNode : public SmNode { protected: SmVisibleNode(SmNodeType eNodeType, const SmToken &rNodeToken) : SmNode(eNodeType, rNodeToken) {} public: virtual bool IsVisible() const override; virtual sal_uInt16 GetNumSubNodes() const override; using SmNode::GetSubNode; virtual SmNode * GetSubNode(sal_uInt16 nIndex) override; }; class SmGraphicNode : public SmVisibleNode { protected: SmGraphicNode(SmNodeType eNodeType, const SmToken &rNodeToken) : SmVisibleNode(eNodeType, rNodeToken) {} public: virtual void GetAccessibleText( OUStringBuffer &rText ) const override; }; /** Draws a rectangle * * Used for drawing the line in the OVER and OVERSTRIKE commands. */ class SmRectangleNode : public SmGraphicNode { Size aToSize; public: SmRectangleNode(const SmToken &rNodeToken) : SmGraphicNode(NRECTANGLE, rNodeToken) {} virtual void AdaptToX(OutputDevice &rDev, sal_uLong nWidth) override; virtual void AdaptToY(OutputDevice &rDev, sal_uLong nHeight) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; }; /** Polygon line node * * Used to draw the slash of the WIDESLASH command by SmBinDiagonalNode. */ class SmPolyLineNode : public SmGraphicNode { tools::Polygon aPoly; Size aToSize; long nWidth; public: SmPolyLineNode(const SmToken &rNodeToken); long GetWidth() const { return nWidth; } tools::Polygon &GetPolygon() { return aPoly; } virtual void AdaptToX(OutputDevice &rDev, sal_uLong nWidth) override; virtual void AdaptToY(OutputDevice &rDev, sal_uLong nHeight) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Text node * * @remarks This class also serves as baseclass for all nodes that contains text. */ class SmTextNode : public SmVisibleNode { OUString aText; sal_uInt16 nFontDesc; /** Index within text where the selection starts * @remarks Only valid if SmNode::IsSelected() is true */ sal_Int32 nSelectionStart; /** Index within text where the selection ends * @remarks Only valid if SmNode::IsSelected() is true */ sal_Int32 nSelectionEnd; protected: SmTextNode(SmNodeType eNodeType, const SmToken &rNodeToken, sal_uInt16 nFontDescP ); public: SmTextNode(const SmToken &rNodeToken, sal_uInt16 nFontDescP ); sal_uInt16 GetFontDesc() const { return nFontDesc; } void SetText(const OUString &rText) { aText = rText; } const OUString & GetText() const { return aText; } /** Change the text of this node, including the underlying token */ void ChangeText(const OUString &rText) { aText = rText; SmToken token = GetToken(); token.aText = rText; SetToken(token); //TODO: Merge this with AdjustFontDesc for better performance AdjustFontDesc(); } /** Try to guess the correct FontDesc, used during visual editing */ void AdjustFontDesc(); /** Index within GetText() where the selection starts * @remarks Only valid of SmNode::IsSelected() is true */ sal_Int32 GetSelectionStart() const {return nSelectionStart;} /** Index within GetText() where the selection end * @remarks Only valid of SmNode::IsSelected() is true */ sal_Int32 GetSelectionEnd() const {return nSelectionEnd;} /** Set the index within GetText() where the selection starts */ void SetSelectionStart(sal_Int32 index) {nSelectionStart = index;} /** Set the index within GetText() where the selection end */ void SetSelectionEnd(sal_Int32 index) {nSelectionEnd = index;} virtual void Prepare(const SmFormat &rFormat, const SmDocShell &rDocShell) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; virtual void CreateTextFromNode(OUString &rText) override; virtual void GetAccessibleText( OUStringBuffer &rText ) const override; void Accept(SmVisitor* pVisitor) override; /** Converts the character from StarMath's private area symbols to a matching Unicode character, if necessary. To be used when converting GetText() to a normal text. */ static sal_Unicode ConvertSymbolToUnicode(sal_Unicode nIn); }; /** Special node for user defined characters * * Node used for pre- and user-defined characters from: * officecfg/registry/data/org/openoffice/Office/Math.xcu * * This is just single characters, I think. */ class SmSpecialNode : public SmTextNode { bool bIsFromGreekSymbolSet; protected: SmSpecialNode(SmNodeType eNodeType, const SmToken &rNodeToken, sal_uInt16 _nFontDesc); public: SmSpecialNode(const SmToken &rNodeToken); virtual void Prepare(const SmFormat &rFormat, const SmDocShell &rDocShell) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Glyph node for custom operators * * This node is used with commands: oper, uoper and boper. * E.g. in "A boper op B", "op" will be an instance of SmGlyphSpecialNode. * "boper" simply interprets "op", the following token, as an binary operator. * The command "uoper" interprets the following token as unary operator. * For these commands an instance of SmGlyphSpecialNode is used for the * operator token, following the command. */ class SmGlyphSpecialNode : public SmSpecialNode { public: SmGlyphSpecialNode(const SmToken &rNodeToken) : SmSpecialNode(NGLYPH_SPECIAL, rNodeToken, FNT_MATH) {} virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Math symbol node * * Use for math symbols such as plus, minus and integral in the INT command. */ class SmMathSymbolNode : public SmSpecialNode { protected: SmMathSymbolNode(SmNodeType eNodeType, const SmToken &rNodeToken) : SmSpecialNode(eNodeType, rNodeToken, FNT_MATH) { sal_Unicode cChar = GetToken().cMathChar; if (sal_Unicode('\0') != cChar) SetText(OUString(cChar)); } public: SmMathSymbolNode(const SmToken &rNodeToken); virtual void AdaptToX(OutputDevice &rDev, sal_uLong nWidth) override; virtual void AdaptToY(OutputDevice &rDev, sal_uLong nHeight) override; virtual void Prepare(const SmFormat &rFormat, const SmDocShell &rDocShell) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; }; /** Math Identifier * * This behaves essentially the same as SmMathSymbolNode and is only used to * represent math symbols that should be exported as elements rather than * elements. */ class SmMathIdentifierNode : public SmMathSymbolNode { public: SmMathIdentifierNode(const SmToken &rNodeToken) : SmMathSymbolNode(NMATHIDENT, rNodeToken) {} }; /** Root symbol node * * Root symbol node used by SmRootNode to create the root symbol, in front of * the line with the line above. I don't think this node should be used for * anything else. */ class SmRootSymbolNode : public SmMathSymbolNode { sal_uLong nBodyWidth; // width of body (argument) of root sign public: SmRootSymbolNode(const SmToken &rNodeToken) : SmMathSymbolNode(NROOTSYMBOL, rNodeToken) , nBodyWidth(0) { } sal_uLong GetBodyWidth() const {return nBodyWidth;}; virtual void AdaptToX(OutputDevice &rDev, sal_uLong nHeight) override; virtual void AdaptToY(OutputDevice &rDev, sal_uLong nHeight) override; void Accept(SmVisitor* pVisitor) override; }; /** Dynamic Integral symbol node * * Node for drawing dynamically sized integral symbols. * * TODO: It might be created a parent class SmDynamicSizedNode (for both dynamic integrals, roots and other dynamic symbols) */ class SmDynIntegralSymbolNode : public SmMathSymbolNode { public: SmDynIntegralSymbolNode(const SmToken &rNodeToken) : SmMathSymbolNode(NDYNINTSYMBOL, rNodeToken) {} virtual void AdaptToY(OutputDevice &rDev, sal_uLong nHeight) override; void Accept(SmVisitor* pVisitor) override; }; /** Place node * * Used to create the command, that denotes place where something can be * written. * It is drawn as a square with a shadow. */ class SmPlaceNode : public SmMathSymbolNode { public: SmPlaceNode(const SmToken &rNodeToken) : SmMathSymbolNode(NPLACE, rNodeToken) { } SmPlaceNode() : SmMathSymbolNode(NPLACE, SmToken(TPLACE, MS_PLACE, "")) {}; virtual void Prepare(const SmFormat &rFormat, const SmDocShell &rDocShell) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Error node, for parsing errors * * This node is used for parsing errors and draws an questionmark turned upside * down (inverted question mark). */ class SmErrorNode : public SmMathSymbolNode { public: SmErrorNode(SmParseError /*eError*/, const SmToken &rNodeToken) : SmMathSymbolNode(NERROR, rNodeToken) { SetText(OUString(MS_ERROR)); } virtual void Prepare(const SmFormat &rFormat, const SmDocShell &rDocShell) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Table node * * This is the root node for the formula tree. This node is also used for the * STACK and BINOM commands. When used for root node, its * children are instances of SmLineNode, and in some obscure cases the a child * can be an instance of SmExpressionNode, mainly when errors occur. */ class SmTableNode : public SmStructureNode { long nFormulaBaseline; public: SmTableNode(const SmToken &rNodeToken) : SmStructureNode(NTABLE, rNodeToken) , nFormulaBaseline(0) { } using SmNode::GetLeftMost; virtual SmNode * GetLeftMost() override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; virtual long GetFormulaBaseline() const override; void Accept(SmVisitor* pVisitor) override; }; /** A line * * Used as child of SmTableNode when the SmTableNode is the root node of the * formula tree. */ class SmLineNode : public SmStructureNode { bool bUseExtraSpaces; protected: SmLineNode(SmNodeType eNodeType, const SmToken &rNodeToken) : SmStructureNode(eNodeType, rNodeToken) { bUseExtraSpaces = true; } public: SmLineNode(const SmToken &rNodeToken) : SmStructureNode(NLINE, rNodeToken) { bUseExtraSpaces = true; } void SetUseExtraSpaces(bool bVal) { bUseExtraSpaces = bVal; } bool IsUseExtraSpaces() const { return bUseExtraSpaces; }; virtual void Prepare(const SmFormat &rFormat, const SmDocShell &rDocShell) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Expression node * * Used whenever you have an expression such as "A OVER {B + C}", here there is * an expression node that allows "B + C" to be the denominator of the * SmBinVerNode, that the OVER command creates. */ class SmExpressionNode : public SmLineNode { public: SmExpressionNode(const SmToken &rNodeToken) : SmLineNode(NEXPRESSION, rNodeToken) {} virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; }; /** Unary horizontal node * * The same as SmBinHorNode except this is for unary operators. */ class SmUnHorNode : public SmStructureNode { public: SmUnHorNode(const SmToken &rNodeToken) : SmStructureNode(NUNHOR, rNodeToken) { SetNumSubNodes(2); } virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Root node * * Used for create square roots and other roots, example: * \f$ \sqrt[\mbox{[Argument]}]{\mbox{[Body]}} \f$. * * Children:
* 0: Argument (optional)
* 1: Symbol (instance of SmRootSymbolNode)
* 2: Body
* Where argument is optional and may be NULL. */ class SmRootNode : public SmStructureNode { protected: static void GetHeightVerOffset(const SmRect &rRect, long &rHeight, long &rVerOffset); static Point GetExtraPos(const SmRect &rRootSymbol, const SmRect &rExtra); public: SmRootNode(const SmToken &rNodeToken) : SmStructureNode(NROOT, rNodeToken) { SetNumSubNodes(3); } virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; SmNode* Argument(); const SmNode* Argument() const; SmRootSymbolNode* Symbol(); const SmRootSymbolNode* Symbol() const; SmNode* Body(); const SmNode* Body() const; }; /** Dynamic Integral node * * Used to create Dynamically sized integrals * * Children:
* 0: Symbol (instance of DynIntegralSymbolNode)
* 1: Body
*/ class SmDynIntegralNode : public SmStructureNode { public: SmDynIntegralNode(const SmToken &rNodeToken) : SmStructureNode(NDYNINT, rNodeToken) { SetNumSubNodes(2); } virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; SmDynIntegralSymbolNode* Symbol(); const SmDynIntegralSymbolNode* Symbol() const; SmNode* Body(); const SmNode* Body() const; }; /** Binary horizontal node * * This node is used for binary operators. In a formula such as "A + B". * * Children:
* 0: Left operand
* 1: Binary operator
* 2: Right operand
* * None of the children may be NULL. */ class SmBinHorNode : public SmStructureNode { public: SmBinHorNode(const SmToken &rNodeToken) : SmStructureNode(NBINHOR, rNodeToken) { SetNumSubNodes(3); } virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; SmMathSymbolNode* Symbol(); const SmMathSymbolNode* Symbol() const; SmNode* LeftOperand(); const SmNode* LeftOperand() const; SmNode* RightOperand(); const SmNode* RightOperand() const; }; /** Binary horizontal node * * This node is used for creating the OVER command, consider the formula: * "numerator OVER denominator", which looks like * \f$ \frac{\mbox{numerator}}{\mbox{denominator}} \f$ * * Children:
* 0: Numerator
* 1: Line (instance of SmRectangleNode)
* 2: Denominator
* None of the children may be NULL. */ class SmBinVerNode : public SmStructureNode { public: SmBinVerNode(const SmToken &rNodeToken) : SmStructureNode(NBINVER, rNodeToken) { SetNumSubNodes(3); } using SmNode::GetLeftMost; virtual SmNode * GetLeftMost() override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; }; /** Binary diagonal node * * Used for implementing the WIDESLASH command, example: "A WIDESLASH B". * * Children:
* 0: Left operand
* 1: right operand
* 2: Line (instance of SmPolyLineNode).
* None of the children may be NULL. */ class SmBinDiagonalNode : public SmStructureNode { bool bAscending; void GetOperPosSize(Point &rPos, Size &rSize, const Point &rDiagPoint, double fAngleDeg) const; public: SmBinDiagonalNode(const SmToken &rNodeToken); bool IsAscending() const { return bAscending; } void SetAscending(bool bVal) { bAscending = bVal; } virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Enum used to index sub-/supscripts in the 'aSubNodes' array * in 'SmSubSupNode' * * See graphic for positions at char: * * \code * CSUP * * LSUP H H RSUP * H H * HHHH * H H * LSUB H H RSUB * * CSUB * \endcode */ enum SmSubSup { CSUB, CSUP, RSUB, RSUP, LSUB, LSUP }; /** numbers of entries in the above enum (that is: the number of possible * sub-/supscripts) */ #define SUBSUP_NUM_ENTRIES 6 /** Super- and subscript node * * Used for creating super- and subscripts for commands such as: * "^", "_", "lsup", "lsub", "csup" and "csub". * Example: "A^2" which looks like: \f$ A^2 \f$ * * This node is also used for creating limits on SmOperNode, when * "FROM" and "TO" commands are used with "INT", "SUM" or similar. * * Children of this node can be enumerated using the SmSubSup enum. * Please note that children may be NULL, except for the body. * It is recommended that you access children using GetBody() and * GetSubSup(). */ class SmSubSupNode : public SmStructureNode { bool bUseLimits; public: SmSubSupNode(const SmToken &rNodeToken) : SmStructureNode(NSUBSUP, rNodeToken) { SetNumSubNodes(1 + SUBSUP_NUM_ENTRIES); bUseLimits = false; } /** Get body (Not NULL) */ SmNode * GetBody() { return GetSubNode(0); } /** Get body (Not NULL) */ const SmNode * GetBody() const { return const_cast(this)->GetBody(); } void SetUseLimits(bool bVal) { bUseLimits = bVal; } bool IsUseLimits() const { return bUseLimits; }; /** Get super- or subscript * @remarks this method may return NULL. */ SmNode * GetSubSup(SmSubSup eSubSup) { return GetSubNode( sal::static_int_cast< sal_uInt16 >(1 + eSubSup) ); }; const SmNode * GetSubSup(SmSubSup eSubSup) const { return const_cast< SmSubSupNode* >( this )->GetSubSup( eSubSup ); } /** Set the body */ void SetBody(SmNode* pBody) { SetSubNode(0, pBody); } void SetSubSup(SmSubSup eSubSup, SmNode* pScript) { SetSubNode( 1 + eSubSup, pScript); } virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; }; /** Node for brace construction * * Used for "lbrace [body] rbrace" and similar constructions. * Should look like \f$ \{\mbox{[body]}\} \f$ * * Children:
* 0: Opening brace
* 1: Body (usually SmBracebodyNode)
* 2: Closing brace
* None of the children can be NULL. * * Note that child 1 (Body) is usually SmBracebodyNode, but it can also be e.g. SmExpressionNode. */ class SmBraceNode : public SmStructureNode { public: SmBraceNode(const SmToken &rNodeToken) : SmStructureNode(NBRACE, rNodeToken) { SetNumSubNodes(3); } SmMathSymbolNode* OpeningBrace(); const SmMathSymbolNode* OpeningBrace() const; SmNode* Body(); const SmNode* Body() const; SmMathSymbolNode* ClosingBrace(); const SmMathSymbolNode* ClosingBrace() const; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; }; /** Body of an SmBraceNode * * This usually only has one child an SmExpressionNode, however, it can also * have other children. * Consider the formula "lbrace [body1] mline [body2] rbrace", looks like: * \f$ \{\mbox{[body1] | [body2]}\} \f$. * In this case SmBracebodyNode will have three children, "[body1]", "|" and * [body2]. */ class SmBracebodyNode : public SmStructureNode { long nBodyHeight; public: inline SmBracebodyNode(const SmToken &rNodeToken); virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; long GetBodyHeight() const { return nBodyHeight; } void Accept(SmVisitor* pVisitor) override; }; inline SmBracebodyNode::SmBracebodyNode(const SmToken &rNodeToken) : SmStructureNode(NBRACEBODY, rNodeToken) { nBodyHeight = 0; } /** Node for vertical brace construction * * Used to implement commands "[body] underbrace [script]" and * "[body] overbrace [script]". * Underbrace should look like this \f$ \underbrace{\mbox{body}}_{\mbox{script}}\f$. * * Children:
* 0: body
* 1: brace
* 2: script
* (None of these children are optional, e.g. they must all be not NULL). */ class SmVerticalBraceNode : public SmStructureNode { public: inline SmVerticalBraceNode(const SmToken &rNodeToken); SmNode* Body(); const SmNode* Body() const; SmMathSymbolNode* Brace(); const SmMathSymbolNode* Brace() const; SmNode* Script(); const SmNode* Script() const; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; inline SmVerticalBraceNode::SmVerticalBraceNode(const SmToken &rNodeToken) : SmStructureNode(NVERTICAL_BRACE, rNodeToken) { SetNumSubNodes(3); } /** Operation Node * * Used for commands like SUM, INT and similar. * * Children:
* 0: Operation (instance of SmMathSymbolNode or SmSubSupNode)
* 1: Body
* None of the children may be NULL. * */ class SmOperNode : public SmStructureNode { public: SmOperNode(const SmToken &rNodeToken) : SmStructureNode(NOPER, rNodeToken) { SetNumSubNodes(2); } SmNode * GetSymbol(); const SmNode * GetSymbol() const { return const_cast(this)->GetSymbol(); } long CalcSymbolHeight(const SmNode &rSymbol, const SmFormat &rFormat) const; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Node used for alignment * * This node has exactly one child at index 0. */ class SmAlignNode : public SmStructureNode { public: SmAlignNode(const SmToken &rNodeToken) : SmStructureNode(NALIGN, rNodeToken) {} virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; /** Attribute node * * Used to give an attribute to another node. Used for commands such as: * UNDERLINE, OVERLINE, OVERSTRIKE, WIDEVEC, WIDEHAT and WIDETILDE. * * Children:
* 0: Attribute
* 1: Body
* None of these may be NULL. */ class SmAttributNode : public SmStructureNode { public: SmAttributNode(const SmToken &rNodeToken) : SmStructureNode(NATTRIBUT, rNodeToken) {} virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; SmNode* Attribute(); const SmNode* Attribute() const; SmNode* Body(); const SmNode* Body() const; }; /** Font node * * Used to change the font of its children. */ class SmFontNode : public SmStructureNode { FontSizeType nSizeType; Fraction aFontSize; public: SmFontNode(const SmToken &rNodeToken) : SmStructureNode(NFONT, rNodeToken) { nSizeType = FontSizeType::MULTIPLY; aFontSize = Fraction(1L); } void SetSizeParameter(const Fraction &rValue, FontSizeType nType); const Fraction & GetSizeParameter() const {return aFontSize;} const FontSizeType& GetSizeType() const {return nSizeType;} virtual void Prepare(const SmFormat &rFormat, const SmDocShell &rDocShell) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; }; /** Matrix node * * Used to implement the MATRIX command, example: * "matrix{ 1 # 2 ## 3 # 4}". */ class SmMatrixNode : public SmStructureNode { sal_uInt16 nNumRows, nNumCols; public: SmMatrixNode(const SmToken &rNodeToken) : SmStructureNode(NMATRIX, rNodeToken) { nNumRows = nNumCols = 0; } sal_uInt16 GetNumRows() const {return nNumRows;} sal_uInt16 GetNumCols() const {return nNumCols;} void SetRowCol(sal_uInt16 nMatrixRows, sal_uInt16 nMatrixCols); using SmNode::GetLeftMost; virtual SmNode * GetLeftMost() override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void CreateTextFromNode(OUString &rText) override; void Accept(SmVisitor* pVisitor) override; }; /** Node for whitespace * * Used to implement the "~" command. This node is just a blank space. */ class SmBlankNode : public SmGraphicNode { sal_uInt16 nNum; public: SmBlankNode(const SmToken &rNodeToken) : SmGraphicNode(NBLANK, rNodeToken) { nNum = 0; } void IncreaseBy(const SmToken &rToken); void Clear() { nNum = 0; } sal_uInt16 GetBlankNum() const { return nNum; } void SetBlankNum(sal_uInt16 nNumber) { nNum = nNumber; } virtual void Prepare(const SmFormat &rFormat, const SmDocShell &rDocShell) override; virtual void Arrange(OutputDevice &rDev, const SmFormat &rFormat) override; void Accept(SmVisitor* pVisitor) override; }; inline SmNode* SmRootNode::Argument() { assert( GetNumSubNodes() == 3 ); return GetSubNode( 0 ); } inline const SmNode* SmRootNode::Argument() const { return const_cast< SmRootNode* >( this )->Argument(); } inline SmRootSymbolNode* SmRootNode::Symbol() { assert( GetNumSubNodes() == 3 ); OSL_ASSERT( GetSubNode( 1 )->GetType() == NROOTSYMBOL ); return static_cast< SmRootSymbolNode* >( GetSubNode( 1 )); } inline const SmRootSymbolNode* SmRootNode::Symbol() const { return const_cast< SmRootNode* >( this )->Symbol(); } inline SmNode* SmRootNode::Body() { assert( GetNumSubNodes() == 3 ); return GetSubNode( 2 ); } inline const SmNode* SmRootNode::Body() const { return const_cast< SmRootNode* >( this )->Body(); } inline SmDynIntegralSymbolNode* SmDynIntegralNode::Symbol() { OSL_ASSERT( GetNumSubNodes() > 0 && GetSubNode( 0 )->GetType() == NDYNINTSYMBOL ); return static_cast< SmDynIntegralSymbolNode* >( GetSubNode( 0 )); } inline const SmDynIntegralSymbolNode* SmDynIntegralNode::Symbol() const { return const_cast< SmDynIntegralNode* >( this )->Symbol(); } inline SmNode* SmDynIntegralNode::Body() { OSL_ASSERT( GetNumSubNodes() > 1 ); return GetSubNode( 1 ); } inline const SmNode* SmDynIntegralNode::Body() const { return const_cast< SmDynIntegralNode* >( this )->Body(); } inline SmMathSymbolNode* SmBinHorNode::Symbol() { OSL_ASSERT( GetNumSubNodes() > 1 && GetSubNode( 1 )->GetType() == NMATH ); return static_cast< SmMathSymbolNode* >( GetSubNode( 1 )); } inline const SmMathSymbolNode* SmBinHorNode::Symbol() const { return const_cast< SmBinHorNode* >( this )->Symbol(); } inline SmNode* SmBinHorNode::LeftOperand() { OSL_ASSERT( GetNumSubNodes() > 0 ); return GetSubNode( 0 ); } inline const SmNode* SmBinHorNode::LeftOperand() const { return const_cast< SmBinHorNode* >( this )->LeftOperand(); } inline SmNode* SmBinHorNode::RightOperand() { OSL_ASSERT( GetNumSubNodes() > 2 ); return GetSubNode( 2 ); } inline const SmNode* SmBinHorNode::RightOperand() const { return const_cast< SmBinHorNode* >( this )->RightOperand(); } inline SmNode* SmAttributNode::Attribute() { OSL_ASSERT( GetNumSubNodes() > 0 ); return GetSubNode( 0 ); } inline const SmNode* SmAttributNode::Attribute() const { return const_cast< SmAttributNode* >( this )->Attribute(); } inline SmNode* SmAttributNode::Body() { OSL_ASSERT( GetNumSubNodes() > 1 ); return GetSubNode( 1 ); } inline const SmNode* SmAttributNode::Body() const { return const_cast< SmAttributNode* >( this )->Body(); } inline SmMathSymbolNode* SmBraceNode::OpeningBrace() { OSL_ASSERT( GetNumSubNodes() > 0 && GetSubNode( 0 )->GetType() == NMATH ); return static_cast< SmMathSymbolNode* >( GetSubNode( 0 )); } inline const SmMathSymbolNode* SmBraceNode::OpeningBrace() const { return const_cast< SmBraceNode* >( this )->OpeningBrace(); } inline SmNode* SmBraceNode::Body() { OSL_ASSERT( GetNumSubNodes() > 1 ); return GetSubNode( 1 ); } inline const SmNode* SmBraceNode::Body() const { return const_cast< SmBraceNode* >( this )->Body(); } inline SmMathSymbolNode* SmBraceNode::ClosingBrace() { OSL_ASSERT( GetNumSubNodes() > 2 && GetSubNode( 2 )->GetType() == NMATH ); return static_cast< SmMathSymbolNode* >( GetSubNode( 2 )); } inline const SmMathSymbolNode* SmBraceNode::ClosingBrace() const { return const_cast< SmBraceNode* >( this )->ClosingBrace(); } inline SmNode* SmVerticalBraceNode::Body() { OSL_ASSERT( GetNumSubNodes() > 0 ); return GetSubNode( 0 ); } inline const SmNode* SmVerticalBraceNode::Body() const { return const_cast< SmVerticalBraceNode* >( this )->Body(); } inline SmMathSymbolNode* SmVerticalBraceNode::Brace() { OSL_ASSERT( GetNumSubNodes() > 1 && GetSubNode( 1 )->GetType() == NMATH ); return static_cast< SmMathSymbolNode* >( GetSubNode( 1 )); } inline const SmMathSymbolNode* SmVerticalBraceNode::Brace() const { return const_cast< SmVerticalBraceNode* >( this )->Brace(); } inline SmNode* SmVerticalBraceNode::Script() { OSL_ASSERT( GetNumSubNodes() > 2 ); return GetSubNode( 2 ); } inline const SmNode* SmVerticalBraceNode::Script() const { return const_cast< SmVerticalBraceNode* >( this )->Script(); } #endif /* vim:set shiftwidth=4 softtabstop=4 expandtab: */