2022-06-08 21:38:26 +08:00
/*
Copyright ( c ) 2008 - 2022 Jan W . Krieger ( < jan @ jkrieger . de > )
with contributions from : Razi Alavizadeh
This software is free software : you can redistribute it and / or modify
it under the terms of the GNU Lesser General Public License ( LGPL ) as published by
the Free Software Foundation , either version 2.1 of the License , or
( at your option ) any later version .
This program 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 ( LGPL ) for more details .
You should have received a copy of the GNU Lesser General Public License ( LGPL )
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# ifndef JKQTMATHTEXTNODE_H
# define JKQTMATHTEXTNODE_H
# include "jkqtmathtext/jkqtmathtext_imexport.h"
# include "jkqtmathtext/jkqtmathtexttools.h"
# include <QPainter>
class JKQTMathText ; // forward
/** \brief subclass representing one node in the syntax tree
* \ ingroup jkqtmathtext_items
*
2022-06-09 20:42:25 +08:00
* \ image html jkqtmathtext / jkqtmathtext_node_geo . png
2022-06-08 21:38:26 +08:00
*/
class JKQTMATHTEXT_LIB_EXPORT JKQTMathTextNode {
public :
2022-06-09 05:52:22 +08:00
explicit JKQTMathTextNode ( JKQTMathText * parentMathText ) ;
2022-06-08 21:38:26 +08:00
JKQTMathTextNode ( const JKQTMathTextNode & ) = delete ;
JKQTMathTextNode & operator = ( const JKQTMathTextNode & ) = delete ;
virtual ~ JKQTMathTextNode ( ) ;
/** \brief determine the size of the node, calls getSizeInternal() implementation of the actual type \see getSizeInternal()
*
* \ param painter painter to use for determining the size
* \ param currentEv current environment object
* \ param [ out ] width width of the block / node
* \ param [ out ] baselineHeight distance from the bottom of the block / node - box to the baseline
* \ param [ out ] overallHeight overall height ( bottom to top ) of the node , the ascent is \ c overallHeight - baselineHeight
* \ param [ out ] strikeoutPos position of the strikeout - line
* \ param [ in ] prevNodeSize optional parameter , describing the size of the previous node ( on the left ) . This may be used for layout of some nodes ( e . g . sub / super to move correctly next to large parantheses . . . )
*
*/
void getSize ( QPainter & painter , JKQTMathTextEnvironment currentEv , double & width , double & baselineHeight , double & overallHeight , double & strikeoutPos , const JKQTMathTextNodeSize * prevNodeSize = nullptr ) ;
2022-08-07 23:48:05 +08:00
/** \brief determine the size of the node, calls getSizeInternal() implementation of the actual type \see getSizeInternal()
*
* \ param painter painter to use for determining the size
* \ param currentEv current environment object
* \ param [ in ] prevNodeSize optional parameter , describing the size of the previous node ( on the left ) . This may be used for layout of some nodes ( e . g . sub / super to move correctly next to large parantheses . . . )
*
*/
JKQTMathTextNodeSize getSize ( QPainter & painter , JKQTMathTextEnvironment currentEv , const JKQTMathTextNodeSize * prevNodeSize = nullptr ) ;
2022-07-24 20:50:28 +08:00
/** \brief calculates the x-size-difference between the given (probably) italic (width externally calculated: \a width_potentiallyitalic, \a ev_potentiallyitalic) and the non-italic version of \a child */
2022-06-08 21:38:26 +08:00
double getNonItalicXCorretion ( QPainter & painter , double width_potentiallyitalic , const JKQTMathTextEnvironment & ev_potentiallyitalic , JKQTMathTextNode * child ) const ;
/** \brief draw the contents at the designated position
*
* \ param painter QPainter to use
* \ param x x - position , where the drawing starts [ Pixel ]
* \ param y Y - position of the baseline , where the drawing starts [ Pixel ]
* \ param currentEv JKQTMathTextEnvironment object describing the current drawing environment / settings
* \ param [ in ] prevNodeSize optional parameter , describing the size of the previous node ( on the left ) . This may be used for layout of some nodes ( e . g . sub / super to move correctly next to large parantheses . . . )
* \ return the x position which to use for the next part of the text
*/
virtual double draw ( QPainter & painter , double x , double y , JKQTMathTextEnvironment currentEv , const JKQTMathTextNodeSize * prevNodeSize = nullptr ) = 0 ;
/** \brief convert node to HTML and returns \c true on success
* \ param [ out ] html new HTML code is APPENDED to this string
* \ param currentEv JKQTMathTextEnvironment object describing the current drawing environment / settings
* \ param defaultEv JKQTMathTextEnvironment object describing the default drawing environment / settings when starting to interpret a node tree
* \ return \ c true on success
*/
virtual bool toHtml ( QString & html , JKQTMathTextEnvironment currentEv , JKQTMathTextEnvironment defaultEv ) ;
/** \brief returns the drawing of colored boxes (for DEBUGGING) around the actual output of the node is enabled */
bool getDrawBoxes ( ) const ;
/** \brief enables the drawing of colored boxes (for DEBUGGING) around the actual output of the node */
virtual void setDrawBoxes ( bool draw ) ;
/** \brief return the name of this class as a string */
virtual QString getTypeName ( ) const ;
2022-06-09 05:52:22 +08:00
/** \copydoc parentNode */
void setParentNode ( JKQTMathTextNode * node ) ;
/** \copydoc parentNode */
JKQTMathTextNode * getParentNode ( ) ;
/** \copydoc parentNode */
const JKQTMathTextNode * getParentNode ( ) const ;
2022-06-26 06:28:49 +08:00
/** \copydoc subSuperscriptAboveBelowNode */
bool isSubSuperscriptAboveBelowNode ( ) const ;
/** \copydoc subSuperscriptAboveBelowNode */
void setSubSuperscriptAboveBelowNode ( bool __value ) ;
2022-06-08 21:38:26 +08:00
protected :
/** \brief determine the size of the node, overwrite this function in derived classes
*
* \ param painter painter to use for determining the size
* \ param currentEv current environment object
* \ param [ out ] width width of the block / node
* \ param [ out ] baselineHeight distance from the bottom of the block / node - box to the baseline
* \ param [ out ] overallHeight overall height ( bottom to top ) of the node , the ascent is \ c overallHeight - baselineHeight
* \ param [ out ] strikeoutPos position of the strikeout - line
* \ param [ in ] prevNodeSize optional parameter , describing the size of the previous node ( on the left ) . This may be used for layout of some nodes ( e . g . sub / super to move correctly next to large parantheses . . . )
*
*/
virtual void getSizeInternal ( QPainter & painter , JKQTMathTextEnvironment currentEv , double & width , double & baselineHeight , double & overallHeight , double & strikeoutPos , const JKQTMathTextNodeSize * prevNodeSize = nullptr ) = 0 ;
/** \brief parent JKQTMathText object (required for several drawing operations */
2022-06-09 05:52:22 +08:00
JKQTMathText * parentMathText ;
/** \brief parent node of this node (i.e. one level up, ode \c nullptr ) */
JKQTMathTextNode * parentNode ;
2022-06-08 21:38:26 +08:00
/** \brief enables the drawing of colored boxes (for DEBUGGING) around the actual output of the node */
bool drawBoxes ;
2022-06-26 06:28:49 +08:00
/** \brief if \c true then following sub- and/or superscripts are placed below and above the node, not besides it.
* This is activated when \ c \ \ sum \ \ limits_ { sub } ^ { sup } is used in LaTeX instead of simply \ c \ \ sum_ { sub } ^ { sup }
*
* The default is c false
*/
bool subSuperscriptAboveBelowNode ;
2022-06-08 21:38:26 +08:00
/** \brief draws colored boxes (for DEBUGGING) around the actual output of the node
*
* \ param painter QPainter to use
* \ param x x - position , where the drawing starts [ Pixel ]
* \ param y Y - position of the baseline , where the drawing starts [ Pixel ]
* \ param currentEv JKQTMathTextEnvironment object describing the current drawing environment / settings
*/
void doDrawBoxes ( QPainter & painter , double x , double y , JKQTMathTextEnvironment currentEv ) ;
2022-06-09 05:52:22 +08:00
/** \brief returns the list of parent, parent-of-parent, ... that can be cast to type \a T */
template < class T >
inline QList < T * > getParents ( ) {
QList < T * > lst ;
JKQTMathTextNode * p = getParentNode ( ) ;
T * pT = dynamic_cast < T * > ( p ) ;
while ( p ! = nullptr ) {
if ( pT ! = nullptr ) lst . append ( pT ) ;
p = p - > getParentNode ( ) ;
pT = dynamic_cast < T * > ( p ) ;
}
return lst ;
}
/** \brief returns the list of parent, parent-of-parent, ... that can be cast to type \a T */
template < class T >
inline QList < const T * > getParents ( ) const {
QList < const T * > lst ;
const JKQTMathTextNode * p = getParentNode ( ) ;
const T * pT = dynamic_cast < const T * > ( p ) ;
while ( p ! = nullptr ) {
if ( pT ! = nullptr ) lst . append ( pT ) ;
p = p - > getParentNode ( ) ;
pT = dynamic_cast < const T * > ( p ) ;
}
return lst ;
}
} ;
2022-06-08 21:38:26 +08:00
2022-06-09 05:52:22 +08:00
/** \brief subclass representing a node in the syntax tree, that has one child
* \ ingroup jkqtmathtext_items
*/
class JKQTMATHTEXT_LIB_EXPORT JKQTMathTextSingleChildNode : public JKQTMathTextNode {
public :
explicit JKQTMathTextSingleChildNode ( JKQTMathTextNode * _child , JKQTMathText * parentMathText ) ;
virtual ~ JKQTMathTextSingleChildNode ( ) override ;
/** \copydoc child */
JKQTMathTextNode * getChild ( ) ;
/** \copydoc child */
const JKQTMathTextNode * getChild ( ) const ;
2022-06-20 04:36:38 +08:00
/** \brief replaces the child node with the node \a newChild , returns the replaced old node */
JKQTMathTextNode * replaceChild ( JKQTMathTextNode * newChild ) ;
/** \brief replaces the child node with the node \a newChild , deletes the replaced old node */
void replaceAndDeleteChild ( JKQTMathTextNode * newChild ) ;
/** \brief returns \c true if the child is valie (\c !=nullptr ) */
bool hasChild ( ) const ;
2022-06-09 05:52:22 +08:00
/** \copydoc JKQTMathTextNode::setDrawBoxes() */
virtual void setDrawBoxes ( bool draw ) override ;
protected :
/** \brief child node of this node */
JKQTMathTextNode * child ;
} ;
2022-06-08 21:38:26 +08:00
2022-06-09 05:52:22 +08:00
/** \brief subclass representing a node in the syntax tree, that has two children
* \ ingroup jkqtmathtext_items
*/
2022-06-20 04:36:38 +08:00
class JKQTMATHTEXT_LIB_EXPORT JKQTMathTextMultiChildNode : public JKQTMathTextNode {
public :
explicit JKQTMathTextMultiChildNode ( JKQTMathText * parentMathText ) ;
virtual ~ JKQTMathTextMultiChildNode ( ) override ;
/** \brief returns a list of all child node */
virtual QList < JKQTMathTextNode * > getChildren ( ) ;
/** \brief returns a list of all child node */
virtual QList < const JKQTMathTextNode * > getChildren ( ) const ;
/** \brief returns the i-th child node */
virtual JKQTMathTextNode * getChild ( int i ) = 0 ;
/** \brief returns the i-th child node */
virtual const JKQTMathTextNode * getChild ( int i ) const = 0 ;
2022-06-26 06:28:49 +08:00
/** \brief returns the first child node */
virtual JKQTMathTextNode * getFirstChild ( ) ;
/** \brief returns the first child node */
virtual const JKQTMathTextNode * getFirstChild ( ) const ;
/** \brief returns the last child node */
virtual JKQTMathTextNode * getLastChild ( ) ;
/** \brief returns the last child node */
virtual const JKQTMathTextNode * getLastChild ( ) const ;
2022-06-20 04:36:38 +08:00
/** \brief replaces the i-th child node with the node \a newChild , returns the replaced old node */
virtual JKQTMathTextNode * replaceChild ( int i , JKQTMathTextNode * newChild ) = 0 ;
/** \brief replaces the i-th child node with the node \a newChild , deletes the replaced old node */
virtual void replaceAndDeleteChild ( int i , JKQTMathTextNode * newChild ) ;
/** \brief returns the number of child nodes */
virtual int childCount ( ) const = 0 ;
2022-06-26 06:28:49 +08:00
/** \brief returns \c true if children exist */
bool hasChildren ( ) const ;
/** \brief returns \c true if there are no children */
bool isEmpty ( ) const ;
2022-06-20 04:36:38 +08:00
/** \brief clear all children, deleting them if \a deleteChildren==true */
virtual void clearChildren ( bool deleteChildren = true ) = 0 ;
2022-08-12 18:54:10 +08:00
/** \brief delete the i-th child */
virtual void deleteChild ( int i ) = 0 ;
2022-06-20 04:36:38 +08:00
/** \copydoc JKQTMathTextNode::setDrawBoxes() */
virtual void setDrawBoxes ( bool draw ) override ;
protected :
} ;
/** \brief subclass representing a node in the syntax tree, that has two children
* \ ingroup jkqtmathtext_items
*/
class JKQTMATHTEXT_LIB_EXPORT JKQTMathTextDualChildNode : public JKQTMathTextMultiChildNode {
2022-06-09 05:52:22 +08:00
public :
explicit JKQTMathTextDualChildNode ( JKQTMathTextNode * _child1 , JKQTMathTextNode * _child2 , JKQTMathText * parentMathText ) ;
virtual ~ JKQTMathTextDualChildNode ( ) override ;
/** \copydoc child1 */
JKQTMathTextNode * getChild1 ( ) ;
/** \copydoc child1 */
const JKQTMathTextNode * getChild1 ( ) const ;
/** \copydoc child2 */
JKQTMathTextNode * getChild2 ( ) ;
/** \copydoc child2 */
const JKQTMathTextNode * getChild2 ( ) const ;
2022-06-20 04:36:38 +08:00
/** \copydoc JKQTMathTextMultiChildNode::getChild() */
virtual JKQTMathTextNode * getChild ( int i ) override ;
/** \copydoc JKQTMathTextMultiChildNode::getChild() */
virtual const JKQTMathTextNode * getChild ( int i ) const override ;
/** \copydoc JKQTMathTextMultiChildNode::getChild() */
virtual JKQTMathTextNode * replaceChild ( int i , JKQTMathTextNode * newChild ) override ;
/** \copydoc JKQTMathTextMultiChildNode::getChild() */
virtual int childCount ( ) const override ;
/** \copydoc JKQTMathTextMultiChildNode::getChild() */
virtual void clearChildren ( bool deleteChildren = true ) override ;
2022-08-12 18:54:10 +08:00
/** \copydoc JKQTMathTextMultiChildNode::deleteChild() */
virtual void deleteChild ( int i ) override ;
2022-06-20 04:36:38 +08:00
2022-06-09 05:52:22 +08:00
protected :
/** \brief first child node of this node */
JKQTMathTextNode * child1 ;
/** \brief second child node of this node */
JKQTMathTextNode * child2 ;
} ;
2022-06-08 21:38:26 +08:00
2022-08-13 21:32:31 +08:00
/** \brief subclass representing a node that outputs nothing
* \ ingroup jkqtmathtext_items
*
*/
class JKQTMATHTEXT_LIB_EXPORT JKQTMathTextNoopNode : public JKQTMathTextNode {
public :
explicit JKQTMathTextNoopNode ( JKQTMathText * parent ) ;
virtual ~ JKQTMathTextNoopNode ( ) override ;
/** \copydoc JKQTMathTextNode::getTypeName() */
virtual QString getTypeName ( ) const override ;
/** \copydoc JKQTMathTextNode::toHtml() */
virtual bool toHtml ( QString & html , JKQTMathTextEnvironment currentEv , JKQTMathTextEnvironment defaultEv ) override ;
/** \copydoc JKQTMathTextNode::draw() */
virtual double draw ( QPainter & painter , double x , double y , JKQTMathTextEnvironment currentEv , const JKQTMathTextNodeSize * prevNodeSize = nullptr ) override ;
protected :
/** \copydoc JKQTMathTextNode::getSizeInternal() */
virtual void getSizeInternal ( QPainter & painter , JKQTMathTextEnvironment currentEv , double & width , double & baselineHeight , double & overallHeight , double & strikeoutPos , const JKQTMathTextNodeSize * prevNodeSize = nullptr ) override ;
} ;
2022-06-09 05:52:22 +08:00
# endif // JKQTMATHTEXTNODE_H