/* * Copyright (c) Meta Platforms, Inc. and affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ #pragma once #ifdef __cplusplus #include #include #include "BitUtils.h" #include "CompactValue.h" #include "YGConfig.h" #include "YGLayout.h" #include "YGStyle.h" #include "YGMacros.h" #include "Yoga-internal.h" YGConfigRef YGConfigGetDefault(); struct YOGA_EXPORT YGNode { using MeasureWithContextFn = YGSize (*)(YGNode*, float, YGMeasureMode, float, YGMeasureMode, void*); using BaselineWithContextFn = float (*)(YGNode*, float, float, void*); using PrintWithContextFn = void (*)(YGNode*, void*); private: static constexpr size_t hasNewLayout_ = 0; static constexpr size_t isReferenceBaseline_ = 1; static constexpr size_t isDirty_ = 2; static constexpr size_t nodeType_ = 3; static constexpr size_t measureUsesContext_ = 4; static constexpr size_t baselineUsesContext_ = 5; static constexpr size_t printUsesContext_ = 6; static constexpr size_t useWebDefaults_ = 7; void* context_ = nullptr; uint8_t flags = 1; uint8_t reserved_ = 0; union { YGMeasureFunc noContext; MeasureWithContextFn withContext; } measure_ = {nullptr}; union { YGBaselineFunc noContext; BaselineWithContextFn withContext; } baseline_ = {nullptr}; union { YGPrintFunc noContext; PrintWithContextFn withContext; } print_ = {nullptr}; YGDirtiedFunc dirtied_ = nullptr; YGStyle style_ = {}; YGLayout layout_ = {}; uint32_t lineIndex_ = 0; YGNodeRef owner_ = nullptr; YGVector children_ = {}; YGConfigRef config_; std::array resolvedDimensions_ = { {YGValueUndefined, YGValueUndefined}}; YGFloatOptional relativePosition( const YGFlexDirection axis, const float axisSize) const; void setMeasureFunc(decltype(measure_)); void setBaselineFunc(decltype(baseline_)); void useWebDefaults() { facebook::yoga::detail::setBooleanData(flags, useWebDefaults_, true); style_.flexDirection() = YGFlexDirectionRow; style_.alignContent() = YGAlignStretch; } // DANGER DANGER DANGER! // If the node assigned to has children, we'd either have to deallocate // them (potentially incorrect) or ignore them (danger of leaks). Only ever // use this after checking that there are no children. // DO NOT CHANGE THE VISIBILITY OF THIS METHOD! YGNode& operator=(YGNode&&) = default; using CompactValue = facebook::yoga::detail::CompactValue; public: YGNode() : YGNode{YGConfigGetDefault()} {} explicit YGNode(const YGConfigRef config) : config_{config} { if (config->useWebDefaults) { useWebDefaults(); } }; ~YGNode() = default; // cleanup of owner/children relationships in YGNodeFree YGNode(YGNode&&); // Does not expose true value semantics, as children are not cloned eagerly. // Should we remove this? YGNode(const YGNode& node) = default; // for RB fabric YGNode(const YGNode& node, YGConfigRef config); // assignment means potential leaks of existing children, or alternatively // freeing unowned memory, double free, or freeing stack memory. YGNode& operator=(const YGNode&) = delete; // Getters void* getContext() const { return context_; } uint8_t& reserved() { return reserved_; } uint8_t reserved() const { return reserved_; } void print(void*); bool getHasNewLayout() const { return facebook::yoga::detail::getBooleanData(flags, hasNewLayout_); } YGNodeType getNodeType() const { return facebook::yoga::detail::getEnumData(flags, nodeType_); } bool hasMeasureFunc() const noexcept { return measure_.noContext != nullptr; } YGSize measure(float, YGMeasureMode, float, YGMeasureMode, void*); bool hasBaselineFunc() const noexcept { return baseline_.noContext != nullptr; } float baseline(float width, float height, void* layoutContext); YGDirtiedFunc getDirtied() const { return dirtied_; } // For Performance reasons passing as reference. YGStyle& getStyle() { return style_; } const YGStyle& getStyle() const { return style_; } // For Performance reasons passing as reference. YGLayout& getLayout() { return layout_; } const YGLayout& getLayout() const { return layout_; } uint32_t getLineIndex() const { return lineIndex_; } bool isReferenceBaseline() { return facebook::yoga::detail::getBooleanData(flags, isReferenceBaseline_); } // returns the YGNodeRef that owns this YGNode. An owner is used to identify // the YogaTree that a YGNode belongs to. This method will return the parent // of the YGNode when a YGNode only belongs to one YogaTree or nullptr when // the YGNode is shared between two or more YogaTrees. YGNodeRef getOwner() const { return owner_; } // Deprecated, use getOwner() instead. YGNodeRef getParent() const { return getOwner(); } const YGVector& getChildren() const { return children_; } // Applies a callback to all children, after cloning them if they are not // owned. template void iterChildrenAfterCloningIfNeeded(T callback, void* cloneContext) { int i = 0; for (YGNodeRef& child : children_) { if (child->getOwner() != this) { child = config_->cloneNode(child, this, i, cloneContext); child->setOwner(this); } i += 1; callback(child, cloneContext); } } YGNodeRef getChild(uint32_t index) const { return children_.at(index); } YGConfigRef getConfig() const { return config_; } bool isDirty() const { return facebook::yoga::detail::getBooleanData(flags, isDirty_); } std::array getResolvedDimensions() const { return resolvedDimensions_; } YGValue getResolvedDimension(int index) const { return resolvedDimensions_[index]; } static CompactValue computeEdgeValueForColumn( const YGStyle::Edges& edges, YGEdge edge, CompactValue defaultValue); static CompactValue computeEdgeValueForRow( const YGStyle::Edges& edges, YGEdge rowEdge, YGEdge edge, CompactValue defaultValue); // Methods related to positions, margin, padding and border YGFloatOptional getLeadingPosition( const YGFlexDirection axis, const float axisSize) const; bool isLeadingPositionDefined(const YGFlexDirection axis) const; bool isTrailingPosDefined(const YGFlexDirection axis) const; YGFloatOptional getTrailingPosition( const YGFlexDirection axis, const float axisSize) const; YGFloatOptional getLeadingMargin( const YGFlexDirection axis, const float widthSize) const; YGFloatOptional getTrailingMargin( const YGFlexDirection axis, const float widthSize) const; float getLeadingBorder(const YGFlexDirection flexDirection) const; float getTrailingBorder(const YGFlexDirection flexDirection) const; YGFloatOptional getLeadingPadding( const YGFlexDirection axis, const float widthSize) const; YGFloatOptional getTrailingPadding( const YGFlexDirection axis, const float widthSize) const; YGFloatOptional getLeadingPaddingAndBorder( const YGFlexDirection axis, const float widthSize) const; YGFloatOptional getTrailingPaddingAndBorder( const YGFlexDirection axis, const float widthSize) const; YGFloatOptional getMarginForAxis( const YGFlexDirection axis, const float widthSize) const; // Setters void setContext(void* context) { context_ = context; } void setPrintFunc(YGPrintFunc printFunc) { print_.noContext = printFunc; facebook::yoga::detail::setBooleanData(flags, printUsesContext_, false); } void setPrintFunc(PrintWithContextFn printFunc) { print_.withContext = printFunc; facebook::yoga::detail::setBooleanData(flags, printUsesContext_, true); } void setPrintFunc(std::nullptr_t) { setPrintFunc(YGPrintFunc{nullptr}); } void setHasNewLayout(bool hasNewLayout) { facebook::yoga::detail::setBooleanData(flags, hasNewLayout_, hasNewLayout); } void setNodeType(YGNodeType nodeType) { return facebook::yoga::detail::setEnumData( flags, nodeType_, nodeType); } void setMeasureFunc(YGMeasureFunc measureFunc); void setMeasureFunc(MeasureWithContextFn); void setMeasureFunc(std::nullptr_t) { return setMeasureFunc(YGMeasureFunc{nullptr}); } void setBaselineFunc(YGBaselineFunc baseLineFunc) { facebook::yoga::detail::setBooleanData(flags, baselineUsesContext_, false); baseline_.noContext = baseLineFunc; } void setBaselineFunc(BaselineWithContextFn baseLineFunc) { facebook::yoga::detail::setBooleanData(flags, baselineUsesContext_, true); baseline_.withContext = baseLineFunc; } void setBaselineFunc(std::nullptr_t) { return setBaselineFunc(YGBaselineFunc{nullptr}); } void setDirtiedFunc(YGDirtiedFunc dirtiedFunc) { dirtied_ = dirtiedFunc; } void setStyle(const YGStyle& style) { style_ = style; } void setLayout(const YGLayout& layout) { layout_ = layout; } void setLineIndex(uint32_t lineIndex) { lineIndex_ = lineIndex; } void setIsReferenceBaseline(bool isReferenceBaseline) { facebook::yoga::detail::setBooleanData( flags, isReferenceBaseline_, isReferenceBaseline); } void setOwner(YGNodeRef owner) { owner_ = owner; } void setChildren(const YGVector& children) { children_ = children; } // TODO: rvalue override for setChildren YG_DEPRECATED void setConfig(YGConfigRef config) { config_ = config; } void setDirty(bool isDirty); void setLayoutLastOwnerDirection(YGDirection direction); void setLayoutComputedFlexBasis(const YGFloatOptional computedFlexBasis); void setLayoutComputedFlexBasisGeneration( uint32_t computedFlexBasisGeneration); void setLayoutMeasuredDimension(float measuredDimension, int index); void setLayoutHadOverflow(bool hadOverflow); void setLayoutDimension(float dimension, int index); void setLayoutDirection(YGDirection direction); void setLayoutMargin(float margin, int index); void setLayoutBorder(float border, int index); void setLayoutPadding(float padding, int index); void setLayoutPosition(float position, int index); void setPosition( const YGDirection direction, const float mainSize, const float crossSize, const float ownerWidth); void setLayoutDoesLegacyFlagAffectsLayout(bool doesLegacyFlagAffectsLayout); void setLayoutDidUseLegacyFlag(bool didUseLegacyFlag); void markDirtyAndPropogateDownwards(); // Other methods YGValue marginLeadingValue(const YGFlexDirection axis) const; YGValue marginTrailingValue(const YGFlexDirection axis) const; YGValue resolveFlexBasisPtr() const; void resolveDimension(); YGDirection resolveDirection(const YGDirection ownerDirection); void clearChildren(); /// Replaces the occurrences of oldChild with newChild void replaceChild(YGNodeRef oldChild, YGNodeRef newChild); void replaceChild(YGNodeRef child, uint32_t index); void insertChild(YGNodeRef child, uint32_t index); /// Removes the first occurrence of child bool removeChild(YGNodeRef child); void removeChild(uint32_t index); void cloneChildrenIfNeeded(void*); void markDirtyAndPropogate(); float resolveFlexGrow() const; float resolveFlexShrink() const; bool isNodeFlexible(); bool didUseLegacyFlag(); bool isLayoutTreeEqualToNode(const YGNode& node) const; void reset(); }; #endif