/** * Copyright (c) Facebook, Inc. and its affiliates. * * This source code is licensed under the MIT license found in the LICENSE * file in the root directory of this source tree. */ #include #include #include #include #include #include #include "YGJTypes.h" using namespace facebook::jni; using namespace std; using facebook::yoga::detail::Log; enum YGStyleInput { LayoutDirection, FlexDirection, Flex, FlexGrow, FlexShrink, FlexBasis, FlexBasisPercent, FlexBasisAuto, FlexWrap, Width, WidthPercent, WidthAuto, MinWidth, MinWidthPercent, MaxWidth, MaxWidthPercent, Height, HeightPercent, HeightAuto, MinHeight, MinHeightPercent, MaxHeight, MaxHeightPercent, JustifyContent, AlignItems, AlignSelf, AlignContent, PositionType, AspectRatio, Overflow, Display, Margin, MarginPercent, MarginAuto, Padding, PaddingPercent, Border, Position, PositionPercent, IsReferenceBaseline, }; class PtrJNodeMap { using JNodeArray = JArrayClass; std::map ptrsToIdxs_; alias_ref javaNodes_; public: PtrJNodeMap() : ptrsToIdxs_{}, javaNodes_{} {} PtrJNodeMap( alias_ref nativePointers, alias_ref javaNodes) : javaNodes_{javaNodes} { auto pin = nativePointers->pinCritical(); auto ptrs = pin.get(); for (size_t i = 0, n = pin.size(); i < n; ++i) { ptrsToIdxs_[(YGNodeRef) ptrs[i]] = i; } } local_ref ref(YGNodeRef node) { auto idx = ptrsToIdxs_.find(node); if (idx == ptrsToIdxs_.end()) { return local_ref{}; } else { return javaNodes_->getElement(idx->second); } } }; struct YGNodeContext { weak_ref* ygNodeJObjectRef{nullptr}; int edgeSetFlag = 0; ~YGNodeContext() { delete ygNodeJObjectRef; } }; const int MARGIN = 1; const int PADDING = 2; const int BORDER = 3; static inline YGNodeContext* ygNodeRefToYGNodeContext(YGNodeRef node) { return reinterpret_cast(node->getContext()); } static inline local_ref YGNodeJobject( YGNodeRef node, void* layoutContext) { if (layoutContext == nullptr) { return (reinterpret_cast*>( ygNodeRefToYGNodeContext(node)->ygNodeJObjectRef)) ->lockLocal(); } else { return reinterpret_cast(layoutContext)->ref(node); } } static void YGTransferLayoutDirection( YGNodeRef node, alias_ref javaNode) { static auto layoutDirectionField = javaNode->getClass()->getField("mLayoutDirection"); javaNode->setFieldValue( layoutDirectionField, static_cast(YGNodeLayoutGetDirection(node))); } static void YGTransferLayoutOutputsRecursive( YGNodeRef root, void* layoutContext) { if (!root->getHasNewLayout()) { return; } auto obj = YGNodeJobject(root, layoutContext); if (!obj) { Log::log( root, YGLogLevelError, nullptr, "Java YGNode was GCed during layout calculation\n"); return; } int edgeSetFlag = ygNodeRefToYGNodeContext(root)->edgeSetFlag; static auto widthField = obj->getClass()->getField("mWidth"); static auto heightField = obj->getClass()->getField("mHeight"); static auto leftField = obj->getClass()->getField("mLeft"); static auto topField = obj->getClass()->getField("mTop"); static auto marginLeftField = obj->getClass()->getField("mMarginLeft"); static auto marginTopField = obj->getClass()->getField("mMarginTop"); static auto marginRightField = obj->getClass()->getField("mMarginRight"); static auto marginBottomField = obj->getClass()->getField("mMarginBottom"); static auto paddingLeftField = obj->getClass()->getField("mPaddingLeft"); static auto paddingTopField = obj->getClass()->getField("mPaddingTop"); static auto paddingRightField = obj->getClass()->getField("mPaddingRight"); static auto paddingBottomField = obj->getClass()->getField("mPaddingBottom"); static auto borderLeftField = obj->getClass()->getField("mBorderLeft"); static auto borderTopField = obj->getClass()->getField("mBorderTop"); static auto borderRightField = obj->getClass()->getField("mBorderRight"); static auto borderBottomField = obj->getClass()->getField("mBorderBottom"); static auto hasNewLayoutField = obj->getClass()->getField("mHasNewLayout"); static auto doesLegacyStretchBehaviour = obj->getClass()->getField( "mDoesLegacyStretchFlagAffectsLayout"); obj->setFieldValue(widthField, YGNodeLayoutGetWidth(root)); obj->setFieldValue(heightField, YGNodeLayoutGetHeight(root)); obj->setFieldValue(leftField, YGNodeLayoutGetLeft(root)); obj->setFieldValue(topField, YGNodeLayoutGetTop(root)); obj->setFieldValue( doesLegacyStretchBehaviour, YGNodeLayoutGetDidLegacyStretchFlagAffectLayout(root)); if ((edgeSetFlag & MARGIN) == MARGIN) { obj->setFieldValue( marginLeftField, YGNodeLayoutGetMargin(root, YGEdgeLeft)); obj->setFieldValue(marginTopField, YGNodeLayoutGetMargin(root, YGEdgeTop)); obj->setFieldValue( marginRightField, YGNodeLayoutGetMargin(root, YGEdgeRight)); obj->setFieldValue( marginBottomField, YGNodeLayoutGetMargin(root, YGEdgeBottom)); } if ((edgeSetFlag & PADDING) == PADDING) { obj->setFieldValue( paddingLeftField, YGNodeLayoutGetPadding(root, YGEdgeLeft)); obj->setFieldValue( paddingTopField, YGNodeLayoutGetPadding(root, YGEdgeTop)); obj->setFieldValue( paddingRightField, YGNodeLayoutGetPadding(root, YGEdgeRight)); obj->setFieldValue( paddingBottomField, YGNodeLayoutGetPadding(root, YGEdgeBottom)); } if ((edgeSetFlag & BORDER) == BORDER) { obj->setFieldValue( borderLeftField, YGNodeLayoutGetBorder(root, YGEdgeLeft)); obj->setFieldValue(borderTopField, YGNodeLayoutGetBorder(root, YGEdgeTop)); obj->setFieldValue( borderRightField, YGNodeLayoutGetBorder(root, YGEdgeRight)); obj->setFieldValue( borderBottomField, YGNodeLayoutGetBorder(root, YGEdgeBottom)); } obj->setFieldValue(hasNewLayoutField, true); YGTransferLayoutDirection(root, obj); root->setHasNewLayout(false); for (uint32_t i = 0; i < YGNodeGetChildCount(root); i++) { YGTransferLayoutOutputsRecursive(YGNodeGetChild(root, i), layoutContext); } } static void YGPrint(YGNodeRef node, void* layoutContext) { if (auto obj = YGNodeJobject(node, layoutContext)) { cout << obj->toString() << endl; } else { Log::log( node, YGLogLevelError, nullptr, "Java YGNode was GCed during layout calculation\n"); } } static float YGJNIBaselineFunc( YGNodeRef node, float width, float height, void* layoutContext) { if (auto obj = YGNodeJobject(node, layoutContext)) { return obj->baseline(width, height); } else { return height; } } static inline YGNodeRef _jlong2YGNodeRef(jlong addr) { return reinterpret_cast(static_cast(addr)); } static inline YGConfigRef _jlong2YGConfigRef(jlong addr) { return reinterpret_cast(static_cast(addr)); } static YGSize YGJNIMeasureFunc( YGNodeRef node, float width, YGMeasureMode widthMode, float height, YGMeasureMode heightMode, void* layoutContext) { if (auto obj = YGNodeJobject(node, layoutContext)) { YGTransferLayoutDirection(node, obj); const auto measureResult = obj->measure(width, widthMode, height, heightMode); static_assert( sizeof(measureResult) == 8, "Expected measureResult to be 8 bytes, or two 32 bit ints"); int32_t wBits = 0xFFFFFFFF & (measureResult >> 32); int32_t hBits = 0xFFFFFFFF & measureResult; const float* measuredWidth = reinterpret_cast(&wBits); const float* measuredHeight = reinterpret_cast(&hBits); return YGSize{*measuredWidth, *measuredHeight}; } else { Log::log( node, YGLogLevelError, nullptr, "Java YGNode was GCed during layout calculation\n"); return YGSize{ widthMode == YGMeasureModeUndefined ? 0 : width, heightMode == YGMeasureModeUndefined ? 0 : height, }; } } static int YGJNILogFunc( const YGConfigRef config, const YGNodeRef node, YGLogLevel level, void* layoutContext, const char* format, va_list args) { int result = vsnprintf(NULL, 0, format, args); std::vector buffer(1 + result); vsnprintf(buffer.data(), buffer.size(), format, args); auto jloggerPtr = static_cast*>(YGConfigGetContext(config)); if (jloggerPtr != nullptr) { if (auto obj = YGNodeJobject(node, layoutContext)) { (*jloggerPtr) ->log( obj, JYogaLogLevel::fromInt(level), Environment::current()->NewStringUTF(buffer.data())); } } return result; } YGNodeContext* createYGNodeContext(alias_ref thiz) { YGNodeContext* ygNodeContext = new YGNodeContext(); ygNodeContext->ygNodeJObjectRef = new weak_ref(make_weak(thiz)); return ygNodeContext; } jlong jni_YGNodeNew(alias_ref thiz) { const YGNodeRef node = YGNodeNew(); node->setContext(createYGNodeContext(thiz)); node->setPrintFunc(YGPrint); return reinterpret_cast(node); } jlong jni_YGNodeNewWithConfig( alias_ref thiz, jlong configPointer, jboolean avoidGlobalJNIRefs) { const YGNodeRef node = YGNodeNewWithConfig(_jlong2YGConfigRef(configPointer)); node->setContext(createYGNodeContext(avoidGlobalJNIRefs ? nullptr : thiz)); return reinterpret_cast(node); } void jni_YGNodeFree(alias_ref, jlong nativePointer) { if (nativePointer == 0) { return; } const YGNodeRef node = _jlong2YGNodeRef(nativePointer); auto context = node->getContext(); if (context != nullptr) { delete reinterpret_cast(node->getContext()); } YGNodeFree(node); } void jni_YGNodeClearChildren(jlong nativePointer) { const YGNodeRef node = _jlong2YGNodeRef(nativePointer); node->clearChildren(); } void jni_YGNodeReset(jlong nativePointer) { const YGNodeRef node = _jlong2YGNodeRef(nativePointer); void* context = node->getContext(); YGNodeReset(node); node->setContext(context); } void jni_YGNodePrint(jlong nativePointer) { const YGNodeRef node = _jlong2YGNodeRef(nativePointer); YGNodePrint( node, (YGPrintOptions)( YGPrintOptionsStyle | YGPrintOptionsLayout | YGPrintOptionsChildren)); } void jni_YGNodeInsertChild( jlong nativePointer, jlong childPointer, jint index) { YGNodeInsertChild( _jlong2YGNodeRef(nativePointer), _jlong2YGNodeRef(childPointer), index); } void jni_YGNodeRemoveChild(jlong nativePointer, jlong childPointer) { YGNodeRemoveChild( _jlong2YGNodeRef(nativePointer), _jlong2YGNodeRef(childPointer)); } void jni_YGNodeSetIsReferenceBaseline( jlong nativePointer, jboolean isReferenceBaseline) { YGNodeSetIsReferenceBaseline( _jlong2YGNodeRef(nativePointer), isReferenceBaseline); } jboolean jni_YGNodeIsReferenceBaseline(jlong nativePointer) { return YGNodeIsReferenceBaseline(_jlong2YGNodeRef(nativePointer)); } void jni_YGNodeCalculateLayout( alias_ref, jlong nativePointer, jfloat width, jfloat height, alias_ref nativePointers, alias_ref> javaNodes) { void* layoutContext = nullptr; auto map = PtrJNodeMap{}; if (nativePointers) { map = PtrJNodeMap{nativePointers, javaNodes}; layoutContext = ↦ } const YGNodeRef root = _jlong2YGNodeRef(nativePointer); YGNodeCalculateLayoutWithContext( root, static_cast(width), static_cast(height), YGNodeStyleGetDirection(_jlong2YGNodeRef(nativePointer)), layoutContext); YGTransferLayoutOutputsRecursive(root, layoutContext); } void jni_YGNodeMarkDirty(jlong nativePointer) { YGNodeMarkDirty(_jlong2YGNodeRef(nativePointer)); } void jni_YGNodeMarkDirtyAndPropogateToDescendants(jlong nativePointer) { YGNodeMarkDirtyAndPropogateToDescendants(_jlong2YGNodeRef(nativePointer)); } jboolean jni_YGNodeIsDirty(jlong nativePointer) { return (jboolean) _jlong2YGNodeRef(nativePointer)->isDirty(); } void jni_YGNodeSetHasMeasureFunc(jlong nativePointer, jboolean hasMeasureFunc) { _jlong2YGNodeRef(nativePointer) ->setMeasureFunc(hasMeasureFunc ? YGJNIMeasureFunc : nullptr); } void jni_YGNodeSetHasBaselineFunc( jlong nativePointer, jboolean hasBaselineFunc) { _jlong2YGNodeRef(nativePointer) ->setBaselineFunc(hasBaselineFunc ? YGJNIBaselineFunc : nullptr); } void jni_YGNodeCopyStyle(jlong dstNativePointer, jlong srcNativePointer) { YGNodeCopyStyle( _jlong2YGNodeRef(dstNativePointer), _jlong2YGNodeRef(srcNativePointer)); } #define YG_NODE_JNI_STYLE_PROP(javatype, type, name) \ javatype jni_YGNodeStyleGet##name(jlong nativePointer) { \ return (javatype) YGNodeStyleGet##name(_jlong2YGNodeRef(nativePointer)); \ } \ \ void jni_YGNodeStyleSet##name(jlong nativePointer, javatype value) { \ YGNodeStyleSet##name( \ _jlong2YGNodeRef(nativePointer), static_cast(value)); \ } #define YG_NODE_JNI_STYLE_UNIT_PROP(name) \ local_ref jni_YGNodeStyleGet##name( \ alias_ref, jlong nativePointer) { \ return JYogaValue::create( \ YGNodeStyleGet##name(_jlong2YGNodeRef(nativePointer))); \ } \ \ void jni_YGNodeStyleSet##name(jlong nativePointer, jfloat value) { \ YGNodeStyleSet##name( \ _jlong2YGNodeRef(nativePointer), static_cast(value)); \ } \ \ void jni_YGNodeStyleSet##name##Percent(jlong nativePointer, jfloat value) { \ YGNodeStyleSet##name##Percent( \ _jlong2YGNodeRef(nativePointer), static_cast(value)); \ } #define YG_NODE_JNI_STYLE_UNIT_PROP_AUTO(name) \ YG_NODE_JNI_STYLE_UNIT_PROP(name) \ void jni_YGNodeStyleSet##name##Auto(jlong nativePointer) { \ YGNodeStyleSet##name##Auto(_jlong2YGNodeRef(nativePointer)); \ } #define YG_NODE_JNI_STYLE_EDGE_PROP(javatype, type, name) \ javatype jni_YGNodeStyleGet##name(jlong nativePointer, jint edge) { \ return (javatype) YGNodeStyleGet##name( \ _jlong2YGNodeRef(nativePointer), static_cast(edge)); \ } \ \ void jni_YGNodeStyleSet##name( \ jlong nativePointer, jint edge, javatype value) { \ YGNodeStyleSet##name( \ _jlong2YGNodeRef(nativePointer), \ static_cast(edge), \ static_cast(value)); \ } #define YG_NODE_JNI_STYLE_EDGE_UNIT_PROP(name) \ local_ref jni_YGNodeStyleGet##name( \ alias_ref, jlong nativePointer, jint edge) { \ return JYogaValue::create(YGNodeStyleGet##name( \ _jlong2YGNodeRef(nativePointer), static_cast(edge))); \ } \ \ void jni_YGNodeStyleSet##name( \ jlong nativePointer, jint edge, jfloat value) { \ YGNodeStyleSet##name( \ _jlong2YGNodeRef(nativePointer), \ static_cast(edge), \ static_cast(value)); \ } \ \ void jni_YGNodeStyleSet##name##Percent( \ jlong nativePointer, jint edge, jfloat value) { \ YGNodeStyleSet##name##Percent( \ _jlong2YGNodeRef(nativePointer), \ static_cast(edge), \ static_cast(value)); \ } #define YG_NODE_JNI_STYLE_EDGE_UNIT_PROP_AUTO(name) \ YG_NODE_JNI_STYLE_EDGE_UNIT_PROP(name) \ void jni_YGNodeStyleSet##name##Auto(jlong nativePointer, jint edge) { \ YGNodeStyleSet##name##Auto( \ _jlong2YGNodeRef(nativePointer), static_cast(edge)); \ } YG_NODE_JNI_STYLE_PROP(jint, YGDirection, Direction); YG_NODE_JNI_STYLE_PROP(jint, YGFlexDirection, FlexDirection); YG_NODE_JNI_STYLE_PROP(jint, YGJustify, JustifyContent); YG_NODE_JNI_STYLE_PROP(jint, YGAlign, AlignItems); YG_NODE_JNI_STYLE_PROP(jint, YGAlign, AlignSelf); YG_NODE_JNI_STYLE_PROP(jint, YGAlign, AlignContent); YG_NODE_JNI_STYLE_PROP(jint, YGPositionType, PositionType); YG_NODE_JNI_STYLE_PROP(jint, YGWrap, FlexWrap); YG_NODE_JNI_STYLE_PROP(jint, YGOverflow, Overflow); YG_NODE_JNI_STYLE_PROP(jint, YGDisplay, Display); jfloat jni_YGNodeStyleGetFlex(jlong nativePointer) { return YGNodeStyleGetFlex(_jlong2YGNodeRef(nativePointer)); } void jni_YGNodeStyleSetFlex(jlong nativePointer, jfloat value) { YGNodeStyleSetFlex( _jlong2YGNodeRef(nativePointer), static_cast(value)); } YG_NODE_JNI_STYLE_PROP(jfloat, float, FlexGrow); YG_NODE_JNI_STYLE_PROP(jfloat, float, FlexShrink); YG_NODE_JNI_STYLE_UNIT_PROP_AUTO(FlexBasis); YG_NODE_JNI_STYLE_EDGE_UNIT_PROP(Position); YG_NODE_JNI_STYLE_UNIT_PROP_AUTO(Width); YG_NODE_JNI_STYLE_UNIT_PROP(MinWidth); YG_NODE_JNI_STYLE_UNIT_PROP(MaxWidth); YG_NODE_JNI_STYLE_UNIT_PROP_AUTO(Height); YG_NODE_JNI_STYLE_UNIT_PROP(MinHeight); YG_NODE_JNI_STYLE_UNIT_PROP(MaxHeight); // Yoga specific properties, not compatible with flexbox specification YG_NODE_JNI_STYLE_PROP(jfloat, float, AspectRatio); jlong jni_YGConfigNew(alias_ref) { return reinterpret_cast(YGConfigNew()); } void jni_YGConfigFree(alias_ref, jlong nativePointer) { const YGConfigRef config = _jlong2YGConfigRef(nativePointer); // unique_ptr will destruct the underlying global_ref, if present. auto context = std::unique_ptr>{ static_cast*>(YGConfigGetContext(config))}; YGConfigFree(config); } void jni_YGConfigSetExperimentalFeatureEnabled( alias_ref, jlong nativePointer, jint feature, jboolean enabled) { const YGConfigRef config = _jlong2YGConfigRef(nativePointer); YGConfigSetExperimentalFeatureEnabled( config, static_cast(feature), enabled); } void jni_YGConfigSetShouldDiffLayoutWithoutLegacyStretchBehaviour( alias_ref, jlong nativePointer, jboolean enabled) { const YGConfigRef config = _jlong2YGConfigRef(nativePointer); YGConfigSetShouldDiffLayoutWithoutLegacyStretchBehaviour(config, enabled); } void jni_YGConfigSetUseWebDefaults( alias_ref, jlong nativePointer, jboolean useWebDefaults) { const YGConfigRef config = _jlong2YGConfigRef(nativePointer); YGConfigSetUseWebDefaults(config, useWebDefaults); } void jni_YGConfigSetPrintTreeFlag( alias_ref, jlong nativePointer, jboolean enable) { const YGConfigRef config = _jlong2YGConfigRef(nativePointer); YGConfigSetPrintTreeFlag(config, enable); } void jni_YGConfigSetPointScaleFactor( alias_ref, jlong nativePointer, jfloat pixelsInPoint) { const YGConfigRef config = _jlong2YGConfigRef(nativePointer); YGConfigSetPointScaleFactor(config, pixelsInPoint); } void jni_YGConfigSetUseLegacyStretchBehaviour( alias_ref, jlong nativePointer, jboolean useLegacyStretchBehaviour) { const YGConfigRef config = _jlong2YGConfigRef(nativePointer); YGConfigSetUseLegacyStretchBehaviour(config, useLegacyStretchBehaviour); } void jni_YGConfigSetLogger( alias_ref, jlong nativePointer, alias_ref logger) { const YGConfigRef config = _jlong2YGConfigRef(nativePointer); auto context = reinterpret_cast*>(YGConfigGetContext(config)); if (logger) { if (context == nullptr) { context = new global_ref{}; YGConfigSetContext(config, context); } *context = make_global(static_ref_cast(logger)); config->setLogger(YGJNILogFunc); } else { if (context != nullptr) { delete context; YGConfigSetContext(config, nullptr); } config->setLogger(nullptr); } } static void YGNodeSetStyleInputs( const YGNodeRef node, float* styleInputs, int size) { const auto end = styleInputs + size; while (styleInputs < end) { auto styleInputKey = static_cast((int) *styleInputs++); switch (styleInputKey) { case LayoutDirection: YGNodeStyleSetDirection(node, static_cast(*styleInputs++)); break; case FlexDirection: YGNodeStyleSetFlexDirection( node, static_cast(*styleInputs++)); break; case Flex: YGNodeStyleSetFlex(node, *styleInputs++); break; case FlexGrow: YGNodeStyleSetFlexGrow(node, *styleInputs++); break; case FlexShrink: YGNodeStyleSetFlexShrink(node, *styleInputs++); break; case FlexBasis: YGNodeStyleSetFlexBasis(node, *styleInputs++); break; case FlexBasisPercent: YGNodeStyleSetFlexBasisPercent(node, *styleInputs++); break; case FlexBasisAuto: YGNodeStyleSetFlexBasisAuto(node); break; case FlexWrap: YGNodeStyleSetFlexWrap(node, static_cast(*styleInputs++)); break; case Width: YGNodeStyleSetWidth(node, *styleInputs++); break; case WidthPercent: YGNodeStyleSetWidthPercent(node, *styleInputs++); break; case WidthAuto: YGNodeStyleSetWidthAuto(node); break; case MinWidth: YGNodeStyleSetMinWidth(node, *styleInputs++); break; case MinWidthPercent: YGNodeStyleSetMinWidthPercent(node, *styleInputs++); break; case MaxWidth: YGNodeStyleSetMaxWidth(node, *styleInputs++); break; case MaxWidthPercent: YGNodeStyleSetMaxWidthPercent(node, *styleInputs++); break; case Height: YGNodeStyleSetHeight(node, *styleInputs++); break; case HeightPercent: YGNodeStyleSetHeightPercent(node, *styleInputs++); break; case HeightAuto: YGNodeStyleSetHeightAuto(node); break; case MinHeight: YGNodeStyleSetMinHeight(node, *styleInputs++); break; case MinHeightPercent: YGNodeStyleSetMinHeightPercent(node, *styleInputs++); break; case MaxHeight: YGNodeStyleSetMaxHeight(node, *styleInputs++); break; case MaxHeightPercent: YGNodeStyleSetMaxHeightPercent(node, *styleInputs++); break; case JustifyContent: YGNodeStyleSetJustifyContent( node, static_cast(*styleInputs++)); break; case AlignItems: YGNodeStyleSetAlignItems(node, static_cast(*styleInputs++)); break; case AlignSelf: YGNodeStyleSetAlignSelf(node, static_cast(*styleInputs++)); break; case AlignContent: YGNodeStyleSetAlignContent(node, static_cast(*styleInputs++)); break; case PositionType: YGNodeStyleSetPositionType( node, static_cast(*styleInputs++)); break; case AspectRatio: YGNodeStyleSetAspectRatio(node, *styleInputs++); break; case Overflow: YGNodeStyleSetOverflow(node, static_cast(*styleInputs++)); break; case Display: YGNodeStyleSetDisplay(node, static_cast(*styleInputs++)); break; case Margin: { float edge = *styleInputs++; float marginValue = *styleInputs++; ygNodeRefToYGNodeContext(node)->edgeSetFlag |= MARGIN; YGNodeStyleSetMargin(node, static_cast(edge), marginValue); break; } case MarginPercent: { float edge = *styleInputs++; float marginPercent = *styleInputs++; ygNodeRefToYGNodeContext(node)->edgeSetFlag |= MARGIN; YGNodeStyleSetMarginPercent( node, static_cast(edge), marginPercent); break; } case MarginAuto: { ygNodeRefToYGNodeContext(node)->edgeSetFlag |= MARGIN; YGNodeStyleSetMarginAuto(node, static_cast(*styleInputs++)); break; } case Padding: { float edge = *styleInputs++; float paddingValue = *styleInputs++; ygNodeRefToYGNodeContext(node)->edgeSetFlag |= PADDING; YGNodeStyleSetPadding(node, static_cast(edge), paddingValue); break; } case PaddingPercent: { float edge = *styleInputs++; float paddingPercent = *styleInputs++; ygNodeRefToYGNodeContext(node)->edgeSetFlag |= PADDING; YGNodeStyleSetPaddingPercent( node, static_cast(edge), paddingPercent); break; } case Border: { float edge = *styleInputs++; float borderValue = *styleInputs++; ygNodeRefToYGNodeContext(node)->edgeSetFlag |= BORDER; YGNodeStyleSetBorder(node, static_cast(edge), borderValue); break; } case Position: { float edge = *styleInputs++; float positionValue = *styleInputs++; YGNodeStyleSetPosition(node, static_cast(edge), positionValue); break; } case PositionPercent: { float edge = *styleInputs++; float positionPercent = *styleInputs++; YGNodeStyleSetPositionPercent( node, static_cast(edge), positionPercent); break; } case IsReferenceBaseline: { YGNodeSetIsReferenceBaseline(node, *styleInputs++ == 1 ? true : false); break; } default: break; } } } void jni_YGNodeSetStyleInputs( alias_ref thiz, jlong nativePointer, alias_ref styleInputs, jint size) { float result[size]; styleInputs->getRegion(0, size, result); YGNodeSetStyleInputs(_jlong2YGNodeRef(nativePointer), result, size); } jint jni_YGNodeGetInstanceCount() { return YGNodeGetInstanceCount(); } local_ref jni_YGNodeStyleGetMargin( alias_ref, jlong nativePointer, jint edge) { YGNodeRef yogaNodeRef = _jlong2YGNodeRef(nativePointer); int edgeSetFlag = ygNodeRefToYGNodeContext(yogaNodeRef)->edgeSetFlag; if ((edgeSetFlag & MARGIN) != MARGIN) { return JYogaValue::create(YGValueUndefined); } return JYogaValue::create( YGNodeStyleGetMargin(yogaNodeRef, static_cast(edge))); } void jni_YGNodeStyleSetMargin(jlong nativePointer, jint edge, jfloat margin) { YGNodeRef yogaNodeRef = _jlong2YGNodeRef(nativePointer); ygNodeRefToYGNodeContext(yogaNodeRef)->edgeSetFlag |= MARGIN; YGNodeStyleSetMargin( yogaNodeRef, static_cast(edge), static_cast(margin)); } void jni_YGNodeStyleSetMarginPercent( jlong nativePointer, jint edge, jfloat percent) { YGNodeRef yogaNodeRef = _jlong2YGNodeRef(nativePointer); ygNodeRefToYGNodeContext(yogaNodeRef)->edgeSetFlag |= MARGIN; YGNodeStyleSetMarginPercent( yogaNodeRef, static_cast(edge), static_cast(percent)); } void jni_YGNodeStyleSetMarginAuto(jlong nativePointer, jint edge) { YGNodeRef yogaNodeRef = _jlong2YGNodeRef(nativePointer); ygNodeRefToYGNodeContext(yogaNodeRef)->edgeSetFlag |= MARGIN; YGNodeStyleSetMarginAuto(yogaNodeRef, static_cast(edge)); } local_ref jni_YGNodeStyleGetPadding( alias_ref, jlong nativePointer, jint edge) { YGNodeRef yogaNodeRef = _jlong2YGNodeRef(nativePointer); int edgeSetFlag = ygNodeRefToYGNodeContext(yogaNodeRef)->edgeSetFlag; if ((edgeSetFlag & PADDING) != PADDING) { return JYogaValue::create(YGValueUndefined); } return JYogaValue::create( YGNodeStyleGetPadding(yogaNodeRef, static_cast(edge))); } void jni_YGNodeStyleSetPadding(jlong nativePointer, jint edge, jfloat padding) { YGNodeRef yogaNodeRef = _jlong2YGNodeRef(nativePointer); ygNodeRefToYGNodeContext(yogaNodeRef)->edgeSetFlag |= PADDING; YGNodeStyleSetPadding( yogaNodeRef, static_cast(edge), static_cast(padding)); } void jni_YGNodeStyleSetPaddingPercent( jlong nativePointer, jint edge, jfloat percent) { YGNodeRef yogaNodeRef = _jlong2YGNodeRef(nativePointer); ygNodeRefToYGNodeContext(yogaNodeRef)->edgeSetFlag |= PADDING; YGNodeStyleSetPaddingPercent( yogaNodeRef, static_cast(edge), static_cast(percent)); } jfloat jni_YGNodeStyleGetBorder(jlong nativePointer, jint edge) { YGNodeRef yogaNodeRef = _jlong2YGNodeRef(nativePointer); int edgeSetFlag = ygNodeRefToYGNodeContext(yogaNodeRef)->edgeSetFlag; if ((edgeSetFlag & BORDER) != BORDER) { return (jfloat) YGUndefined; } return (jfloat) YGNodeStyleGetBorder(yogaNodeRef, static_cast(edge)); } void jni_YGNodeStyleSetBorder(jlong nativePointer, jint edge, jfloat border) { YGNodeRef yogaNodeRef = _jlong2YGNodeRef(nativePointer); ygNodeRefToYGNodeContext(yogaNodeRef)->edgeSetFlag |= BORDER; YGNodeStyleSetBorder( yogaNodeRef, static_cast(edge), static_cast(border)); } #define YGMakeNativeMethod(name) makeNativeMethod(#name, name) #define YGMakeCriticalNativeMethod(name) \ makeCriticalNativeMethod_DO_NOT_USE_OR_YOU_WILL_BE_FIRED(#name, name) jint JNI_OnLoad(JavaVM* vm, void*) { return initialize(vm, [] { JYogaNode::javaClassStatic()->registerNatives({ YGMakeNativeMethod(jni_YGNodeNew), YGMakeNativeMethod(jni_YGNodeNewWithConfig), YGMakeNativeMethod(jni_YGNodeFree), YGMakeCriticalNativeMethod(jni_YGNodeReset), YGMakeCriticalNativeMethod(jni_YGNodeClearChildren), YGMakeCriticalNativeMethod(jni_YGNodeInsertChild), YGMakeCriticalNativeMethod(jni_YGNodeRemoveChild), YGMakeCriticalNativeMethod(jni_YGNodeSetIsReferenceBaseline), YGMakeCriticalNativeMethod(jni_YGNodeIsReferenceBaseline), YGMakeNativeMethod(jni_YGNodeCalculateLayout), YGMakeCriticalNativeMethod(jni_YGNodeMarkDirty), YGMakeCriticalNativeMethod( jni_YGNodeMarkDirtyAndPropogateToDescendants), YGMakeCriticalNativeMethod(jni_YGNodeIsDirty), YGMakeCriticalNativeMethod(jni_YGNodeSetHasMeasureFunc), YGMakeCriticalNativeMethod(jni_YGNodeSetHasBaselineFunc), YGMakeCriticalNativeMethod(jni_YGNodeCopyStyle), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetDirection), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetDirection), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetFlexDirection), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetFlexDirection), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetJustifyContent), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetJustifyContent), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetAlignItems), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetAlignItems), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetAlignSelf), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetAlignSelf), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetAlignContent), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetAlignContent), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetPositionType), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetPositionType), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetFlexWrap), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetFlexWrap), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetOverflow), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetOverflow), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetDisplay), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetDisplay), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetFlex), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetFlex), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetFlexGrow), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetFlexGrow), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetFlexShrink), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetFlexShrink), YGMakeNativeMethod(jni_YGNodeStyleGetFlexBasis), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetFlexBasis), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetFlexBasisPercent), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetFlexBasisAuto), YGMakeNativeMethod(jni_YGNodeStyleGetMargin), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMargin), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMarginPercent), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMarginAuto), YGMakeNativeMethod(jni_YGNodeStyleGetPadding), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetPadding), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetPaddingPercent), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetBorder), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetBorder), YGMakeNativeMethod(jni_YGNodeStyleGetPosition), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetPosition), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetPositionPercent), YGMakeNativeMethod(jni_YGNodeStyleGetWidth), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetWidth), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetWidthPercent), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetWidthAuto), YGMakeNativeMethod(jni_YGNodeStyleGetHeight), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetHeight), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetHeightPercent), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetHeightAuto), YGMakeNativeMethod(jni_YGNodeStyleGetMinWidth), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMinWidth), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMinWidthPercent), YGMakeNativeMethod(jni_YGNodeStyleGetMinHeight), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMinHeight), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMinHeightPercent), YGMakeNativeMethod(jni_YGNodeStyleGetMaxWidth), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMaxWidth), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMaxWidthPercent), YGMakeNativeMethod(jni_YGNodeStyleGetMaxHeight), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMaxHeight), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetMaxHeightPercent), YGMakeCriticalNativeMethod(jni_YGNodeStyleGetAspectRatio), YGMakeCriticalNativeMethod(jni_YGNodeStyleSetAspectRatio), YGMakeCriticalNativeMethod(jni_YGNodeGetInstanceCount), YGMakeCriticalNativeMethod(jni_YGNodePrint), YGMakeNativeMethod(jni_YGNodeSetStyleInputs), }); registerNatives( "com/facebook/yoga/YogaConfig", { YGMakeNativeMethod(jni_YGConfigNew), YGMakeNativeMethod(jni_YGConfigFree), YGMakeNativeMethod(jni_YGConfigSetExperimentalFeatureEnabled), YGMakeNativeMethod(jni_YGConfigSetUseWebDefaults), YGMakeNativeMethod(jni_YGConfigSetPrintTreeFlag), YGMakeNativeMethod(jni_YGConfigSetPointScaleFactor), YGMakeNativeMethod(jni_YGConfigSetUseLegacyStretchBehaviour), YGMakeNativeMethod(jni_YGConfigSetLogger), YGMakeNativeMethod( jni_YGConfigSetShouldDiffLayoutWithoutLegacyStretchBehaviour), }); }); }