Fix issue where percentages were off of the border box, not padding box (#1485)
Summary: Pull Request resolved: https://github.com/facebook/yoga/pull/1485 X-link: https://github.com/facebook/react-native/pull/41686 The size of the containing block is the size of the padding box of the containing node for absolute nodes. We were looking at `containingNode->getLayout().measuredDimension(Dimension::Width)` which is the border box. So we need to subtract the border from this. Added a test that was failing before this change as well Reviewed By: NickGerleman Differential Revision: D51330526 fbshipit-source-id: adc448dfb71b54f1bbed0d9d61c5553bda4b106c
This commit is contained in:
committed by
Facebook GitHub Bot
parent
f6c4a8e8e4
commit
9b87d8b3f3
@@ -93,7 +93,7 @@
|
|||||||
<div style="width:20%; height:20%; left:20%; top:20%; position: absolute;"></div>
|
<div style="width:20%; height:20%; left:20%; top:20%; position: absolute;"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="absolute_layout_percentage_height_based_on_padded_parent" data-disabled="true" style="flex-direction:column; width:100px; height:100px; padding-top: 10px; border-top: 10px solid black;">
|
<div id="absolute_layout_percentage_height_based_on_padded_parent" style="flex-direction:column; width:100px; height:100px; padding-top: 10px; border-top: 10px solid black;">
|
||||||
<div style="width:100px; height:50%; position: absolute;"></div>
|
<div style="width:100px; height:50%; position: absolute;"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@@ -324,3 +324,13 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div id="static_position_containing_block_padding_and_border">
|
||||||
|
<div
|
||||||
|
style="width: 400px; height: 400px; padding: 8px 1px 4px 9px; border-width: 5px 7px 4px 2px; position: relative">
|
||||||
|
<div style="height:100px; width: 100px; position: static">
|
||||||
|
<div style="height: 61%; width: 41%; position: absolute">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
@@ -2653,6 +2653,85 @@ public class YGStaticPositionTest {
|
|||||||
assertEquals(50f, root_child0_child0.getLayoutHeight(), 0.0f);
|
assertEquals(50f, root_child0_child0.getLayoutHeight(), 0.0f);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test_static_position_containing_block_padding_and_border() {
|
||||||
|
YogaConfig config = YogaConfigFactory.create();
|
||||||
|
config.setExperimentalFeatureEnabled(YogaExperimentalFeature.ABSOLUTE_PERCENTAGE_AGAINST_PADDING_EDGE, true);
|
||||||
|
|
||||||
|
final YogaNode root = createNode(config);
|
||||||
|
root.setPositionType(YogaPositionType.ABSOLUTE);
|
||||||
|
|
||||||
|
final YogaNode root_child0 = createNode(config);
|
||||||
|
root_child0.setPadding(YogaEdge.LEFT, 9);
|
||||||
|
root_child0.setPadding(YogaEdge.TOP, 8);
|
||||||
|
root_child0.setPadding(YogaEdge.RIGHT, 1);
|
||||||
|
root_child0.setPadding(YogaEdge.BOTTOM, 4);
|
||||||
|
root_child0.setBorder(YogaEdge.LEFT, 2f);
|
||||||
|
root_child0.setBorder(YogaEdge.TOP, 5f);
|
||||||
|
root_child0.setBorder(YogaEdge.RIGHT, 7f);
|
||||||
|
root_child0.setBorder(YogaEdge.BOTTOM, 4f);
|
||||||
|
root_child0.setWidth(400f);
|
||||||
|
root_child0.setHeight(400f);
|
||||||
|
root.addChildAt(root_child0, 0);
|
||||||
|
|
||||||
|
final YogaNode root_child0_child0 = createNode(config);
|
||||||
|
root_child0_child0.setPositionType(YogaPositionType.STATIC);
|
||||||
|
root_child0_child0.setWidth(100f);
|
||||||
|
root_child0_child0.setHeight(100f);
|
||||||
|
root_child0.addChildAt(root_child0_child0, 0);
|
||||||
|
|
||||||
|
final YogaNode root_child0_child0_child0 = createNode(config);
|
||||||
|
root_child0_child0_child0.setPositionType(YogaPositionType.ABSOLUTE);
|
||||||
|
root_child0_child0_child0.setWidthPercent(41f);
|
||||||
|
root_child0_child0_child0.setHeightPercent(61f);
|
||||||
|
root_child0_child0.addChildAt(root_child0_child0_child0, 0);
|
||||||
|
root.setDirection(YogaDirection.LTR);
|
||||||
|
root.calculateLayout(YogaConstants.UNDEFINED, YogaConstants.UNDEFINED);
|
||||||
|
|
||||||
|
assertEquals(0f, root.getLayoutX(), 0.0f);
|
||||||
|
assertEquals(0f, root.getLayoutY(), 0.0f);
|
||||||
|
assertEquals(400f, root.getLayoutWidth(), 0.0f);
|
||||||
|
assertEquals(400f, root.getLayoutHeight(), 0.0f);
|
||||||
|
|
||||||
|
assertEquals(0f, root_child0.getLayoutX(), 0.0f);
|
||||||
|
assertEquals(0f, root_child0.getLayoutY(), 0.0f);
|
||||||
|
assertEquals(400f, root_child0.getLayoutWidth(), 0.0f);
|
||||||
|
assertEquals(400f, root_child0.getLayoutHeight(), 0.0f);
|
||||||
|
|
||||||
|
assertEquals(11f, root_child0_child0.getLayoutX(), 0.0f);
|
||||||
|
assertEquals(13f, root_child0_child0.getLayoutY(), 0.0f);
|
||||||
|
assertEquals(100f, root_child0_child0.getLayoutWidth(), 0.0f);
|
||||||
|
assertEquals(100f, root_child0_child0.getLayoutHeight(), 0.0f);
|
||||||
|
|
||||||
|
assertEquals(0f, root_child0_child0_child0.getLayoutX(), 0.0f);
|
||||||
|
assertEquals(0f, root_child0_child0_child0.getLayoutY(), 0.0f);
|
||||||
|
assertEquals(160f, root_child0_child0_child0.getLayoutWidth(), 0.0f);
|
||||||
|
assertEquals(239f, root_child0_child0_child0.getLayoutHeight(), 0.0f);
|
||||||
|
|
||||||
|
root.setDirection(YogaDirection.RTL);
|
||||||
|
root.calculateLayout(YogaConstants.UNDEFINED, YogaConstants.UNDEFINED);
|
||||||
|
|
||||||
|
assertEquals(0f, root.getLayoutX(), 0.0f);
|
||||||
|
assertEquals(0f, root.getLayoutY(), 0.0f);
|
||||||
|
assertEquals(400f, root.getLayoutWidth(), 0.0f);
|
||||||
|
assertEquals(400f, root.getLayoutHeight(), 0.0f);
|
||||||
|
|
||||||
|
assertEquals(0f, root_child0.getLayoutX(), 0.0f);
|
||||||
|
assertEquals(0f, root_child0.getLayoutY(), 0.0f);
|
||||||
|
assertEquals(400f, root_child0.getLayoutWidth(), 0.0f);
|
||||||
|
assertEquals(400f, root_child0.getLayoutHeight(), 0.0f);
|
||||||
|
|
||||||
|
assertEquals(292f, root_child0_child0.getLayoutX(), 0.0f);
|
||||||
|
assertEquals(13f, root_child0_child0.getLayoutY(), 0.0f);
|
||||||
|
assertEquals(100f, root_child0_child0.getLayoutWidth(), 0.0f);
|
||||||
|
assertEquals(100f, root_child0_child0.getLayoutHeight(), 0.0f);
|
||||||
|
|
||||||
|
assertEquals(-60f, root_child0_child0_child0.getLayoutX(), 0.0f);
|
||||||
|
assertEquals(0f, root_child0_child0_child0.getLayoutY(), 0.0f);
|
||||||
|
assertEquals(160f, root_child0_child0_child0.getLayoutWidth(), 0.0f);
|
||||||
|
assertEquals(239f, root_child0_child0_child0.getLayoutHeight(), 0.0f);
|
||||||
|
}
|
||||||
|
|
||||||
private YogaNode createNode(YogaConfig config) {
|
private YogaNode createNode(YogaConfig config) {
|
||||||
return mNodeFactory.create(config);
|
return mNodeFactory.create(config);
|
||||||
}
|
}
|
||||||
|
@@ -2874,3 +2874,88 @@ test('static_position_static_child_containing_block_content_box', () => {
|
|||||||
config.free();
|
config.free();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
test('static_position_containing_block_padding_and_border', () => {
|
||||||
|
const config = Yoga.Config.create();
|
||||||
|
let root;
|
||||||
|
|
||||||
|
config.setExperimentalFeatureEnabled(ExperimentalFeature.AbsolutePercentageAgainstPaddingEdge, true);
|
||||||
|
|
||||||
|
try {
|
||||||
|
root = Yoga.Node.create(config);
|
||||||
|
root.setPositionType(PositionType.Absolute);
|
||||||
|
|
||||||
|
const root_child0 = Yoga.Node.create(config);
|
||||||
|
root_child0.setPadding(Edge.Left, 9);
|
||||||
|
root_child0.setPadding(Edge.Top, 8);
|
||||||
|
root_child0.setPadding(Edge.Right, 1);
|
||||||
|
root_child0.setPadding(Edge.Bottom, 4);
|
||||||
|
root_child0.setBorder(Edge.Left, 2);
|
||||||
|
root_child0.setBorder(Edge.Top, 5);
|
||||||
|
root_child0.setBorder(Edge.Right, 7);
|
||||||
|
root_child0.setBorder(Edge.Bottom, 4);
|
||||||
|
root_child0.setWidth(400);
|
||||||
|
root_child0.setHeight(400);
|
||||||
|
root.insertChild(root_child0, 0);
|
||||||
|
|
||||||
|
const root_child0_child0 = Yoga.Node.create(config);
|
||||||
|
root_child0_child0.setPositionType(PositionType.Static);
|
||||||
|
root_child0_child0.setWidth(100);
|
||||||
|
root_child0_child0.setHeight(100);
|
||||||
|
root_child0.insertChild(root_child0_child0, 0);
|
||||||
|
|
||||||
|
const root_child0_child0_child0 = Yoga.Node.create(config);
|
||||||
|
root_child0_child0_child0.setPositionType(PositionType.Absolute);
|
||||||
|
root_child0_child0_child0.setWidth("41%");
|
||||||
|
root_child0_child0_child0.setHeight("61%");
|
||||||
|
root_child0_child0.insertChild(root_child0_child0_child0, 0);
|
||||||
|
root.calculateLayout(undefined, undefined, Direction.LTR);
|
||||||
|
|
||||||
|
expect(root.getComputedLeft()).toBe(0);
|
||||||
|
expect(root.getComputedTop()).toBe(0);
|
||||||
|
expect(root.getComputedWidth()).toBe(400);
|
||||||
|
expect(root.getComputedHeight()).toBe(400);
|
||||||
|
|
||||||
|
expect(root_child0.getComputedLeft()).toBe(0);
|
||||||
|
expect(root_child0.getComputedTop()).toBe(0);
|
||||||
|
expect(root_child0.getComputedWidth()).toBe(400);
|
||||||
|
expect(root_child0.getComputedHeight()).toBe(400);
|
||||||
|
|
||||||
|
expect(root_child0_child0.getComputedLeft()).toBe(11);
|
||||||
|
expect(root_child0_child0.getComputedTop()).toBe(13);
|
||||||
|
expect(root_child0_child0.getComputedWidth()).toBe(100);
|
||||||
|
expect(root_child0_child0.getComputedHeight()).toBe(100);
|
||||||
|
|
||||||
|
expect(root_child0_child0_child0.getComputedLeft()).toBe(0);
|
||||||
|
expect(root_child0_child0_child0.getComputedTop()).toBe(0);
|
||||||
|
expect(root_child0_child0_child0.getComputedWidth()).toBe(160);
|
||||||
|
expect(root_child0_child0_child0.getComputedHeight()).toBe(239);
|
||||||
|
|
||||||
|
root.calculateLayout(undefined, undefined, Direction.RTL);
|
||||||
|
|
||||||
|
expect(root.getComputedLeft()).toBe(0);
|
||||||
|
expect(root.getComputedTop()).toBe(0);
|
||||||
|
expect(root.getComputedWidth()).toBe(400);
|
||||||
|
expect(root.getComputedHeight()).toBe(400);
|
||||||
|
|
||||||
|
expect(root_child0.getComputedLeft()).toBe(0);
|
||||||
|
expect(root_child0.getComputedTop()).toBe(0);
|
||||||
|
expect(root_child0.getComputedWidth()).toBe(400);
|
||||||
|
expect(root_child0.getComputedHeight()).toBe(400);
|
||||||
|
|
||||||
|
expect(root_child0_child0.getComputedLeft()).toBe(292);
|
||||||
|
expect(root_child0_child0.getComputedTop()).toBe(13);
|
||||||
|
expect(root_child0_child0.getComputedWidth()).toBe(100);
|
||||||
|
expect(root_child0_child0.getComputedHeight()).toBe(100);
|
||||||
|
|
||||||
|
expect(root_child0_child0_child0.getComputedLeft()).toBe(-60);
|
||||||
|
expect(root_child0_child0_child0.getComputedTop()).toBe(0);
|
||||||
|
expect(root_child0_child0_child0.getComputedWidth()).toBe(160);
|
||||||
|
expect(root_child0_child0_child0.getComputedHeight()).toBe(239);
|
||||||
|
} finally {
|
||||||
|
if (typeof root !== 'undefined') {
|
||||||
|
root.freeRecursive();
|
||||||
|
}
|
||||||
|
|
||||||
|
config.free();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
@@ -2676,3 +2676,83 @@ TEST(YogaTest, static_position_static_child_containing_block_content_box) {
|
|||||||
|
|
||||||
YGConfigFree(config);
|
YGConfigFree(config);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(YogaTest, static_position_containing_block_padding_and_border) {
|
||||||
|
const YGConfigRef config = YGConfigNew();
|
||||||
|
YGConfigSetExperimentalFeatureEnabled(config, YGExperimentalFeatureAbsolutePercentageAgainstPaddingEdge, true);
|
||||||
|
|
||||||
|
const YGNodeRef root = YGNodeNewWithConfig(config);
|
||||||
|
YGNodeStyleSetPositionType(root, YGPositionTypeAbsolute);
|
||||||
|
|
||||||
|
const YGNodeRef root_child0 = YGNodeNewWithConfig(config);
|
||||||
|
YGNodeStyleSetPadding(root_child0, YGEdgeLeft, 9);
|
||||||
|
YGNodeStyleSetPadding(root_child0, YGEdgeTop, 8);
|
||||||
|
YGNodeStyleSetPadding(root_child0, YGEdgeRight, 1);
|
||||||
|
YGNodeStyleSetPadding(root_child0, YGEdgeBottom, 4);
|
||||||
|
YGNodeStyleSetBorder(root_child0, YGEdgeLeft, 2);
|
||||||
|
YGNodeStyleSetBorder(root_child0, YGEdgeTop, 5);
|
||||||
|
YGNodeStyleSetBorder(root_child0, YGEdgeRight, 7);
|
||||||
|
YGNodeStyleSetBorder(root_child0, YGEdgeBottom, 4);
|
||||||
|
YGNodeStyleSetWidth(root_child0, 400);
|
||||||
|
YGNodeStyleSetHeight(root_child0, 400);
|
||||||
|
YGNodeInsertChild(root, root_child0, 0);
|
||||||
|
|
||||||
|
const YGNodeRef root_child0_child0 = YGNodeNewWithConfig(config);
|
||||||
|
YGNodeStyleSetPositionType(root_child0_child0, YGPositionTypeStatic);
|
||||||
|
YGNodeStyleSetWidth(root_child0_child0, 100);
|
||||||
|
YGNodeStyleSetHeight(root_child0_child0, 100);
|
||||||
|
YGNodeInsertChild(root_child0, root_child0_child0, 0);
|
||||||
|
|
||||||
|
const YGNodeRef root_child0_child0_child0 = YGNodeNewWithConfig(config);
|
||||||
|
YGNodeStyleSetPositionType(root_child0_child0_child0, YGPositionTypeAbsolute);
|
||||||
|
YGNodeStyleSetWidthPercent(root_child0_child0_child0, 41);
|
||||||
|
YGNodeStyleSetHeightPercent(root_child0_child0_child0, 61);
|
||||||
|
YGNodeInsertChild(root_child0_child0, root_child0_child0_child0, 0);
|
||||||
|
YGNodeCalculateLayout(root, YGUndefined, YGUndefined, YGDirectionLTR);
|
||||||
|
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetLeft(root));
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetTop(root));
|
||||||
|
ASSERT_FLOAT_EQ(400, YGNodeLayoutGetWidth(root));
|
||||||
|
ASSERT_FLOAT_EQ(400, YGNodeLayoutGetHeight(root));
|
||||||
|
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetLeft(root_child0));
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetTop(root_child0));
|
||||||
|
ASSERT_FLOAT_EQ(400, YGNodeLayoutGetWidth(root_child0));
|
||||||
|
ASSERT_FLOAT_EQ(400, YGNodeLayoutGetHeight(root_child0));
|
||||||
|
|
||||||
|
ASSERT_FLOAT_EQ(11, YGNodeLayoutGetLeft(root_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(13, YGNodeLayoutGetTop(root_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(100, YGNodeLayoutGetWidth(root_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(100, YGNodeLayoutGetHeight(root_child0_child0));
|
||||||
|
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetLeft(root_child0_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetTop(root_child0_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(160, YGNodeLayoutGetWidth(root_child0_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(239, YGNodeLayoutGetHeight(root_child0_child0_child0));
|
||||||
|
|
||||||
|
YGNodeCalculateLayout(root, YGUndefined, YGUndefined, YGDirectionRTL);
|
||||||
|
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetLeft(root));
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetTop(root));
|
||||||
|
ASSERT_FLOAT_EQ(400, YGNodeLayoutGetWidth(root));
|
||||||
|
ASSERT_FLOAT_EQ(400, YGNodeLayoutGetHeight(root));
|
||||||
|
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetLeft(root_child0));
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetTop(root_child0));
|
||||||
|
ASSERT_FLOAT_EQ(400, YGNodeLayoutGetWidth(root_child0));
|
||||||
|
ASSERT_FLOAT_EQ(400, YGNodeLayoutGetHeight(root_child0));
|
||||||
|
|
||||||
|
ASSERT_FLOAT_EQ(292, YGNodeLayoutGetLeft(root_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(13, YGNodeLayoutGetTop(root_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(100, YGNodeLayoutGetWidth(root_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(100, YGNodeLayoutGetHeight(root_child0_child0));
|
||||||
|
|
||||||
|
ASSERT_FLOAT_EQ(-60, YGNodeLayoutGetLeft(root_child0_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(0, YGNodeLayoutGetTop(root_child0_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(160, YGNodeLayoutGetWidth(root_child0_child0_child0));
|
||||||
|
ASSERT_FLOAT_EQ(239, YGNodeLayoutGetHeight(root_child0_child0_child0));
|
||||||
|
|
||||||
|
YGNodeFreeRecursive(root);
|
||||||
|
|
||||||
|
YGConfigFree(config);
|
||||||
|
}
|
||||||
|
@@ -25,7 +25,8 @@ static void justifyAbsoluteChild(
|
|||||||
case Justify::SpaceBetween:
|
case Justify::SpaceBetween:
|
||||||
child->setLayoutPosition(
|
child->setLayoutPosition(
|
||||||
child->getFlexStartMargin(mainAxis, direction, containingBlockWidth) +
|
child->getFlexStartMargin(mainAxis, direction, containingBlockWidth) +
|
||||||
parent->getFlexStartBorder(mainAxis, direction),
|
parent->getLayout().border(flexStartEdge(mainAxis)) +
|
||||||
|
parent->getLayout().padding(flexStartEdge(mainAxis)),
|
||||||
flexStartEdge(mainAxis));
|
flexStartEdge(mainAxis));
|
||||||
break;
|
break;
|
||||||
case Justify::FlexEnd:
|
case Justify::FlexEnd:
|
||||||
@@ -458,8 +459,10 @@ void layoutAbsoluteDescendants(
|
|||||||
containingNode,
|
containingNode,
|
||||||
currentNode,
|
currentNode,
|
||||||
child,
|
child,
|
||||||
containingNode->getLayout().measuredDimension(Dimension::Width),
|
containingNode->getLayout().measuredDimension(Dimension::Width) -
|
||||||
containingNode->getLayout().measuredDimension(Dimension::Height),
|
containingNode->getBorderForAxis(FlexDirection::Row),
|
||||||
|
containingNode->getLayout().measuredDimension(Dimension::Height) -
|
||||||
|
containingNode->getBorderForAxis(FlexDirection::Column),
|
||||||
widthSizingMode,
|
widthSizingMode,
|
||||||
currentNodeDirection,
|
currentNodeDirection,
|
||||||
layoutMarkerData,
|
layoutMarkerData,
|
||||||
|
@@ -377,6 +377,11 @@ float Node::getFlexEndPaddingAndBorder(
|
|||||||
getFlexEndBorder(axis, direction);
|
getFlexEndBorder(axis, direction);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float Node::getBorderForAxis(FlexDirection axis) const {
|
||||||
|
return getInlineStartBorder(axis, Direction::LTR) +
|
||||||
|
getInlineEndBorder(axis, Direction::LTR);
|
||||||
|
}
|
||||||
|
|
||||||
float Node::getMarginForAxis(FlexDirection axis, float widthSize) const {
|
float Node::getMarginForAxis(FlexDirection axis, float widthSize) const {
|
||||||
// The total margin for a given axis does not depend on the direction
|
// The total margin for a given axis does not depend on the direction
|
||||||
// so hardcoding LTR here to avoid piping direction to this function
|
// so hardcoding LTR here to avoid piping direction to this function
|
||||||
|
@@ -287,6 +287,7 @@ class YG_EXPORT Node : public ::YGNode {
|
|||||||
FlexDirection axis,
|
FlexDirection axis,
|
||||||
Direction direction,
|
Direction direction,
|
||||||
float widthSize) const;
|
float widthSize) const;
|
||||||
|
float getBorderForAxis(FlexDirection axis) const;
|
||||||
float getMarginForAxis(FlexDirection axis, float widthSize) const;
|
float getMarginForAxis(FlexDirection axis, float widthSize) const;
|
||||||
float getGapForAxis(FlexDirection axis) const;
|
float getGapForAxis(FlexDirection axis) const;
|
||||||
// Setters
|
// Setters
|
||||||
|
Reference in New Issue
Block a user