mirror of
https://github.com/NativeScript/NativeScript.git
synced 2025-08-15 11:01:21 +08:00

* chore: move tns-core-modules to nativescript-core * chore: preparing compat generate script * chore: add missing definitions * chore: no need for http-request to be private * chore: packages chore * test: generate tests for tns-core-modules * chore: add anroid module for consistency * chore: add .npmignore * chore: added privateModulesWhitelist * chore(webpack): added bundle-entry-points * chore: scripts * chore: tests changed to use @ns/core * test: add scoped-packages test project * test: fix types * test: update test project * chore: build scripts * chore: update build script * chore: npm scripts cleanup * chore: make the compat pgk work with old wp config * test: generate diff friendly tests * chore: create barrel exports * chore: move files after rebase * chore: typedoc config * chore: compat mode * chore: review of barrels * chore: remove tns-core-modules import after rebase * chore: dev workflow setup * chore: update developer-workflow * docs: experiment with API extractor * chore: api-extractor and barrel exports * chore: api-extractor configs * chore: generate d.ts rollup with api-extractor * refactor: move methods inside Frame * chore: fic tests to use Frame static methods * refactor: create Builder class * refactor: use Builder class in tests * refactor: include Style in ui barrel * chore: separate compat build script * chore: fix tslint errors * chore: update NATIVESCRIPT_CORE_ARGS * chore: fix compat pack * chore: fix ui-test-app build with linked modules * chore: Application, ApplicationSettings, Connectivity and Http * chore: export Trace, Profiling and Utils * refactor: Static create methods for ImageSource * chore: fix deprecated usages of ImageSource * chore: move Span and FormattedString to ui * chore: add events-args and ImageSource to index files * chore: check for CLI >= 6.2 when building for IOS * chore: update travis build * chore: copy Pod file to compat package * chore: update error msg ui-tests-app * refactor: Apply suggestions from code review Co-Authored-By: Martin Yankov <m.i.yankov@gmail.com> * chore: typings and refs * chore: add missing d.ts files for public API * chore: adress code review FB * chore: update api-report * chore: dev-workflow for other apps * chore: api update * chore: update api-report
1374 lines
63 KiB
TypeScript
1374 lines
63 KiB
TypeScript
import {
|
|
FlexDirection, FlexWrap, JustifyContent, AlignItems, AlignContent,
|
|
FlexboxLayoutBase, View, layout,
|
|
FlexBasisPercent,
|
|
orderProperty, flexGrowProperty, flexShrinkProperty, flexWrapBeforeProperty, alignSelfProperty
|
|
} from "./flexbox-layout-common";
|
|
|
|
export * from "./flexbox-layout-common";
|
|
|
|
import EXACTLY = layout.EXACTLY;
|
|
import AT_MOST = layout.AT_MOST;
|
|
import UNSPECIFIED = layout.UNSPECIFIED;
|
|
|
|
import MEASURED_SIZE_MASK = layout.MEASURED_SIZE_MASK;
|
|
import MEASURED_STATE_TOO_SMALL = layout.MEASURED_STATE_TOO_SMALL;
|
|
|
|
function requestFlexboxLayout(this: View, value) {
|
|
let flexbox = this.parent;
|
|
if (flexbox instanceof FlexboxLayoutBase) {
|
|
flexbox.requestLayout();
|
|
}
|
|
}
|
|
View.prototype[orderProperty.setNative] = requestFlexboxLayout;
|
|
View.prototype[flexGrowProperty.setNative] = requestFlexboxLayout;
|
|
View.prototype[flexShrinkProperty.setNative] = requestFlexboxLayout;
|
|
View.prototype[flexWrapBeforeProperty.setNative] = requestFlexboxLayout;
|
|
View.prototype[alignSelfProperty.setNative] = requestFlexboxLayout;
|
|
|
|
const MATCH_PARENT = -1;
|
|
const WRAP_CONTENT = -2;
|
|
|
|
const View_sUseZeroUnspecifiedMeasureSpec = true; // NOTE: android version < M
|
|
|
|
// Long ints may not be safe in JavaScript
|
|
const MAX_SIZE = 0x00FFFFFF & MEASURED_SIZE_MASK;
|
|
|
|
import makeMeasureSpec = layout.makeMeasureSpec;
|
|
import getMeasureSpecMode = layout.getMeasureSpecMode;
|
|
import getMeasureSpecSize = layout.getMeasureSpecSize;
|
|
|
|
// `eachLayoutChild` iterates over children, and we need more - indexed access.
|
|
// This class tries to accomodate that by collecting all children in an
|
|
// array no more than once per measure.
|
|
class MeasureContext {
|
|
private children: View[];
|
|
|
|
constructor(private owner: FlexboxLayout) {
|
|
this.children = [];
|
|
this.owner.eachLayoutChild((child) => {
|
|
this.children.push(child);
|
|
});
|
|
}
|
|
|
|
public get childrenCount(): number {
|
|
return this.children.length;
|
|
}
|
|
|
|
public childAt(index: number): View {
|
|
return this.children[index];
|
|
}
|
|
}
|
|
|
|
class FlexLine {
|
|
|
|
_left: number = Number.MAX_VALUE;
|
|
_top: number = Number.MAX_VALUE;
|
|
_right: number = Number.MAX_VALUE;
|
|
_bottom: number = Number.MAX_VALUE;
|
|
|
|
_mainSize: number = 0;
|
|
_dividerLengthInMainSize = 0;
|
|
_crossSize: number = 0;
|
|
_itemCount: number = 0;
|
|
_totalFlexGrow: number = 0;
|
|
_totalFlexShrink: number = 0;
|
|
_maxBaseline: number = 0;
|
|
|
|
_indicesAlignSelfStretch: number[] = [];
|
|
|
|
get left(): number { return this._left; }
|
|
get top(): number { return this._top; }
|
|
get right(): number { return this._right; }
|
|
get bottom(): number { return this._bottom; }
|
|
|
|
get mainSize(): number { return this._mainSize; }
|
|
get crossSize(): number { return this._crossSize; }
|
|
get itemCount(): number { return this._itemCount; }
|
|
get totalFlexGrow(): number { return this._totalFlexGrow; }
|
|
get totalFlexShrink(): number { return this._totalFlexShrink; }
|
|
}
|
|
|
|
class Order {
|
|
index: number;
|
|
order: number;
|
|
|
|
public compareTo(another: Order): number {
|
|
if (this.order !== another.order) {
|
|
return this.order - another.order;
|
|
}
|
|
|
|
return this.index - another.index;
|
|
}
|
|
}
|
|
|
|
export class FlexboxLayout extends FlexboxLayoutBase {
|
|
|
|
// Omit divider
|
|
|
|
private _reorderedIndices: number[];
|
|
|
|
private _orderCache: number[];
|
|
private _flexLines: FlexLine[] = [];
|
|
private _childrenFrozen: boolean[];
|
|
private measureContext: MeasureContext;
|
|
|
|
public onMeasure(widthMeasureSpec: number, heightMeasureSpec: number): void {
|
|
this.measureContext = new MeasureContext(this);
|
|
// Omit: super.onMeasure(widthMeasureSpec, heightMeasureSpec);
|
|
|
|
if (this._isOrderChangedFromLastMeasurement) {
|
|
this._reorderedIndices = this._createReorderedIndices();
|
|
}
|
|
if (!this._childrenFrozen || this._childrenFrozen.length < this.measureContext.childrenCount) {
|
|
this._childrenFrozen = new Array(this.measureContext.childrenCount);
|
|
}
|
|
|
|
switch (this.flexDirection) {
|
|
case FlexDirection.ROW:
|
|
case FlexDirection.ROW_REVERSE:
|
|
this._measureHorizontal(widthMeasureSpec, heightMeasureSpec);
|
|
break;
|
|
case FlexDirection.COLUMN:
|
|
case FlexDirection.COLUMN_REVERSE:
|
|
this._measureVertical(widthMeasureSpec, heightMeasureSpec);
|
|
break;
|
|
default:
|
|
throw new Error("Invalid value for the flex direction is set: " + this.flexDirection);
|
|
}
|
|
|
|
this._childrenFrozen.length = 0;
|
|
}
|
|
|
|
private _getReorderedChildAt(index: number): View {
|
|
let child: View;
|
|
if (index < 0 || index >= this._reorderedIndices.length) {
|
|
child = null;
|
|
} else {
|
|
let reorderedIndex = this._reorderedIndices[index];
|
|
child = this.measureContext.childAt(reorderedIndex);
|
|
}
|
|
|
|
return child;
|
|
}
|
|
|
|
private _createReorderedIndices(): number[] {
|
|
let childCount = this.measureContext.childrenCount;
|
|
let orders = this._createOrders(childCount);
|
|
|
|
return this._sortOrdersIntoReorderedIndices(childCount, orders);
|
|
}
|
|
|
|
private _sortOrdersIntoReorderedIndices(childCount: number, orders: Order[]): number[] {
|
|
orders.sort((a, b) => a.compareTo(b));
|
|
if (!this._orderCache) {
|
|
this._orderCache = [];
|
|
}
|
|
|
|
this._orderCache.length = 0;
|
|
let reorderedIndices: number[] = [];
|
|
orders.forEach((order, i) => {
|
|
reorderedIndices[i] = order.index;
|
|
this._orderCache[i] = order.order;
|
|
});
|
|
|
|
return reorderedIndices;
|
|
}
|
|
|
|
private _createOrders(childCount: number): Order[] {
|
|
let orders: Order[] = [];
|
|
for (let i = 0; i < childCount; i++) {
|
|
let child = this.measureContext.childAt(i);
|
|
let order = new Order();
|
|
order.order = FlexboxLayout.getOrder(child);
|
|
order.index = i;
|
|
orders.push(order);
|
|
}
|
|
|
|
return orders;
|
|
}
|
|
|
|
private get _isOrderChangedFromLastMeasurement(): boolean {
|
|
let childCount = this.measureContext.childrenCount;
|
|
if (!this._orderCache) {
|
|
this._orderCache = [];
|
|
}
|
|
if (this._orderCache.length !== childCount) {
|
|
return true;
|
|
}
|
|
for (let i = 0; i < childCount; i++) {
|
|
let view = this.measureContext.childAt(i);
|
|
if (view === null) {
|
|
continue;
|
|
}
|
|
|
|
if (FlexboxLayout.getOrder(view) !== this._orderCache[i]) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private _measureHorizontal(widthMeasureSpec: number, heightMeasureSpec: number): void {
|
|
const widthSize = getMeasureSpecSize(widthMeasureSpec);
|
|
const widthMode = getMeasureSpecMode(widthMeasureSpec);
|
|
const heightSize = getMeasureSpecSize(heightMeasureSpec);
|
|
const heightMode = getMeasureSpecMode(heightMeasureSpec);
|
|
|
|
let childState = 0;
|
|
|
|
this._flexLines.length = 0;
|
|
|
|
(() => {
|
|
let childCount = this.measureContext.childrenCount;
|
|
let paddingStart = FlexboxLayout.getPaddingStart(this);
|
|
let paddingEnd = FlexboxLayout.getPaddingEnd(this);
|
|
let largestHeightInRow = Number.MIN_VALUE;
|
|
let flexLine = new FlexLine();
|
|
|
|
let indexInFlexLine = 0;
|
|
flexLine._mainSize = paddingStart + paddingEnd;
|
|
for (let i = 0; i < childCount; i++) {
|
|
let child = this._getReorderedChildAt(i);
|
|
if (child === null) {
|
|
this._addFlexLineIfLastFlexItem(i, childCount, flexLine);
|
|
continue;
|
|
} else if (child.isCollapsed) {
|
|
flexLine._itemCount++;
|
|
this._addFlexLineIfLastFlexItem(i, childCount, flexLine);
|
|
continue;
|
|
}
|
|
|
|
child._updateEffectiveLayoutValues(widthSize, widthMode, heightSize, heightMode);
|
|
let lp = child; // child.style;
|
|
if (FlexboxLayout.getAlignSelf(child) === "stretch") {
|
|
flexLine._indicesAlignSelfStretch.push(i);
|
|
}
|
|
|
|
let childWidth = lp.effectiveWidth;
|
|
if (FlexBasisPercent.DEFAULT /*lp.flexBasisPercent*/ !== FlexBasisPercent.DEFAULT && widthMode === EXACTLY) {
|
|
childWidth = Math.round(widthSize * FlexBasisPercent.DEFAULT /*lp.flexBasisPercent*/);
|
|
}
|
|
|
|
let childWidthMeasureSpec = FlexboxLayout.getChildMeasureSpec(widthMeasureSpec,
|
|
lp.effectivePaddingLeft + lp.effectivePaddingRight + lp.effectiveMarginLeft
|
|
+ lp.effectiveMarginRight, childWidth < 0 ? WRAP_CONTENT : childWidth);
|
|
|
|
let childHeightMeasureSpec = FlexboxLayout.getChildMeasureSpec(heightMeasureSpec,
|
|
lp.effectivePaddingTop + lp.effectivePaddingBottom + lp.effectiveMarginTop
|
|
+ lp.effectiveMarginBottom, lp.effectiveHeight < 0 ? WRAP_CONTENT : lp.effectiveHeight);
|
|
|
|
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
|
|
|
|
this._checkSizeConstraints(child);
|
|
|
|
childState = View.combineMeasuredStates(childState, child.getMeasuredState());
|
|
largestHeightInRow = Math.max(largestHeightInRow,
|
|
child.getMeasuredHeight() + lp.effectiveMarginTop + lp.effectiveMarginBottom);
|
|
|
|
if (this._isWrapRequired(child, widthMode, widthSize, flexLine._mainSize,
|
|
child.getMeasuredWidth() + lp.effectiveMarginLeft + lp.effectiveMarginRight,
|
|
i, indexInFlexLine)) {
|
|
if (flexLine.itemCount > 0) {
|
|
this._addFlexLine(flexLine);
|
|
}
|
|
|
|
flexLine = new FlexLine();
|
|
flexLine._itemCount = 1;
|
|
flexLine._mainSize = paddingStart + paddingEnd;
|
|
largestHeightInRow = child.getMeasuredHeight() + lp.effectiveMarginTop + lp.effectiveMarginBottom;
|
|
indexInFlexLine = 0;
|
|
} else {
|
|
flexLine._itemCount++;
|
|
indexInFlexLine++;
|
|
}
|
|
flexLine._mainSize += child.getMeasuredWidth() + lp.effectiveMarginLeft + lp.effectiveMarginRight;
|
|
flexLine._totalFlexGrow += FlexboxLayout.getFlexGrow(child);
|
|
flexLine._totalFlexShrink += FlexboxLayout.getFlexShrink(child);
|
|
|
|
flexLine._crossSize = Math.max(flexLine._crossSize, largestHeightInRow);
|
|
|
|
// Omit divider
|
|
|
|
if (this.flexWrap !== FlexWrap.WRAP_REVERSE) {
|
|
flexLine._maxBaseline = Math.max(flexLine._maxBaseline, FlexboxLayout.getBaseline(child) + lp.effectiveMarginTop);
|
|
} else {
|
|
flexLine._maxBaseline = Math.max(flexLine._maxBaseline, child.getMeasuredHeight() - FlexboxLayout.getBaseline(child) + lp.effectiveMarginBottom);
|
|
}
|
|
this._addFlexLineIfLastFlexItem(i, childCount, flexLine);
|
|
}
|
|
})();
|
|
|
|
this._determineMainSize(this.flexDirection, widthMeasureSpec, heightMeasureSpec);
|
|
|
|
if (this.alignItems === AlignItems.BASELINE) {
|
|
let viewIndex = 0;
|
|
this._flexLines.forEach(flexLine => {
|
|
let largestHeightInLine = Number.MIN_VALUE;
|
|
for (let i = viewIndex; i < viewIndex + flexLine._itemCount; i++) {
|
|
let child = this._getReorderedChildAt(i);
|
|
const lp = child; // .style;
|
|
if (this.flexWrap !== FlexWrap.WRAP_REVERSE) {
|
|
let marginTop = flexLine._maxBaseline - FlexboxLayout.getBaseline(child);
|
|
marginTop = Math.max(marginTop, lp.effectiveMarginTop);
|
|
largestHeightInLine = Math.max(largestHeightInLine, child.getActualSize().height + marginTop + lp.effectiveMarginBottom);
|
|
} else {
|
|
let marginBottom = flexLine._maxBaseline - child.getMeasuredHeight() + FlexboxLayout.getBaseline(child);
|
|
marginBottom = Math.max(marginBottom, lp.effectiveMarginBottom);
|
|
largestHeightInLine = Math.max(largestHeightInLine, child.getActualSize().height + lp.effectiveMarginTop + marginBottom);
|
|
}
|
|
}
|
|
flexLine._crossSize = largestHeightInLine;
|
|
viewIndex += flexLine.itemCount;
|
|
});
|
|
}
|
|
|
|
this._determineCrossSize(this.flexDirection, widthMeasureSpec, heightMeasureSpec, this.effectivePaddingTop + this.effectivePaddingBottom);
|
|
this._stretchViews(this.flexDirection, this.alignItems);
|
|
this._setMeasuredDimensionForFlex(this.flexDirection, widthMeasureSpec, heightMeasureSpec, childState);
|
|
}
|
|
|
|
private _measureVertical(widthMeasureSpec, heightMeasureSpec): void {
|
|
const widthSize = getMeasureSpecSize(widthMeasureSpec);
|
|
const widthMode = getMeasureSpecMode(widthMeasureSpec);
|
|
const heightSize = getMeasureSpecSize(heightMeasureSpec);
|
|
const heightMode = getMeasureSpecMode(heightMeasureSpec);
|
|
let childState = 0;
|
|
|
|
this._flexLines.length = 0;
|
|
|
|
let childCount = this.measureContext.childrenCount;
|
|
let paddingTop = this.effectivePaddingTop;
|
|
let paddingBottom = this.effectivePaddingBottom;
|
|
let largestWidthInColumn = Number.MIN_VALUE;
|
|
let flexLine = new FlexLine();
|
|
flexLine._mainSize = paddingTop + paddingBottom;
|
|
|
|
let indexInFlexLine = 0;
|
|
for (let i = 0; i < childCount; i++) {
|
|
let child = this._getReorderedChildAt(i);
|
|
if (child === null) {
|
|
this._addFlexLineIfLastFlexItem(i, childCount, flexLine);
|
|
continue;
|
|
} else if (child.isCollapsed) {
|
|
flexLine._itemCount++;
|
|
this._addFlexLineIfLastFlexItem(i, childCount, flexLine);
|
|
continue;
|
|
}
|
|
|
|
child._updateEffectiveLayoutValues(widthSize, widthMode, heightSize, heightMode);
|
|
const lp = child; // .style;
|
|
if (FlexboxLayout.getAlignSelf(child) === "stretch") {
|
|
flexLine._indicesAlignSelfStretch.push(i);
|
|
}
|
|
|
|
let childHeight = lp.effectiveHeight;
|
|
// TODO: This should always be false
|
|
if (FlexBasisPercent.DEFAULT /* lp.flexBasisPercent */ !== FlexBasisPercent.DEFAULT && heightMode === EXACTLY) {
|
|
childHeight = Math.round(heightSize * FlexBasisPercent.DEFAULT /* lp.flexBasisPercent */);
|
|
}
|
|
|
|
let childWidthMeasureSpec = FlexboxLayout.getChildMeasureSpec(widthMeasureSpec,
|
|
this.effectivePaddingLeft + this.effectivePaddingRight + lp.effectiveMarginLeft
|
|
+ lp.effectiveMarginRight, lp.effectiveWidth < 0 ? WRAP_CONTENT : lp.effectiveWidth);
|
|
let childHeightMeasureSpec = FlexboxLayout.getChildMeasureSpec(heightMeasureSpec,
|
|
this.effectivePaddingTop + this.effectivePaddingBottom + lp.effectiveMarginTop
|
|
+ lp.effectiveMarginBottom, childHeight < 0 ? WRAP_CONTENT : childHeight);
|
|
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
|
|
|
|
this._checkSizeConstraints(child);
|
|
|
|
childState = View.combineMeasuredStates(childState, child.getMeasuredState());
|
|
largestWidthInColumn = Math.max(largestWidthInColumn,
|
|
child.getMeasuredWidth() + lp.effectiveMarginLeft + lp.effectiveMarginRight);
|
|
|
|
if (this._isWrapRequired(child, heightMode, heightSize, flexLine.mainSize,
|
|
child.getMeasuredHeight() + lp.effectiveMarginTop + lp.effectiveMarginBottom,
|
|
i, indexInFlexLine)) {
|
|
|
|
if (flexLine._itemCount > 0) {
|
|
this._addFlexLine(flexLine);
|
|
}
|
|
|
|
flexLine = new FlexLine();
|
|
flexLine._itemCount = 1;
|
|
flexLine._mainSize = paddingTop + paddingBottom;
|
|
largestWidthInColumn = child.getMeasuredWidth() + lp.effectiveMarginLeft + lp.effectiveMarginRight;
|
|
indexInFlexLine = 0;
|
|
} else {
|
|
flexLine._itemCount++;
|
|
indexInFlexLine++;
|
|
}
|
|
|
|
flexLine._mainSize += child.getMeasuredHeight() + lp.effectiveMarginTop + lp.effectiveMarginBottom;
|
|
flexLine._totalFlexGrow += FlexboxLayout.getFlexGrow(child);
|
|
flexLine._totalFlexShrink += FlexboxLayout.getFlexShrink(child);
|
|
|
|
flexLine._crossSize = Math.max(flexLine._crossSize, largestWidthInColumn);
|
|
|
|
// Omit divider
|
|
this._addFlexLineIfLastFlexItem(i, childCount, flexLine);
|
|
}
|
|
|
|
this._determineMainSize(this.flexDirection, widthMeasureSpec, heightMeasureSpec);
|
|
this._determineCrossSize(this.flexDirection, widthMeasureSpec, heightMeasureSpec, this.effectivePaddingLeft + this.effectivePaddingRight);
|
|
this._stretchViews(this.flexDirection, this.alignItems);
|
|
this._setMeasuredDimensionForFlex(this.flexDirection, widthMeasureSpec, heightMeasureSpec, childState);
|
|
}
|
|
|
|
private _checkSizeConstraints(view: View) {
|
|
let needsMeasure = false;
|
|
let childWidth = view.getMeasuredWidth();
|
|
let childHeight = view.getMeasuredHeight();
|
|
|
|
let minWidth = view.effectiveMinWidth;
|
|
view.effectiveMinWidth = 0;
|
|
if (view.getMeasuredWidth() < minWidth) {
|
|
needsMeasure = true;
|
|
childWidth = minWidth;
|
|
} else if (view.getMeasuredWidth() > MAX_SIZE /*lp.maxWidth*/) {
|
|
needsMeasure = true;
|
|
childWidth = MAX_SIZE /*lp.maxWidth*/;
|
|
}
|
|
|
|
let minHeight = view.effectiveMinHeight;
|
|
view.effectiveMinHeight = 0;
|
|
if (childHeight < minHeight) {
|
|
needsMeasure = true;
|
|
childHeight = minHeight;
|
|
} else if (childHeight > MAX_SIZE /*lp.maxWidth*/) {
|
|
needsMeasure = true;
|
|
childHeight = MAX_SIZE /*lp.maxWidth*/;
|
|
}
|
|
if (needsMeasure) {
|
|
view.measure(makeMeasureSpec(childWidth, EXACTLY), makeMeasureSpec(childHeight, EXACTLY));
|
|
}
|
|
view.effectiveMinWidth = minWidth;
|
|
view.effectiveMinHeight = minHeight;
|
|
}
|
|
|
|
private _addFlexLineIfLastFlexItem(childIndex: number, childCount: number, flexLine: FlexLine) {
|
|
if (childIndex === childCount - 1 && flexLine.itemCount !== 0) {
|
|
this._addFlexLine(flexLine);
|
|
}
|
|
}
|
|
|
|
private _addFlexLine(flexLine: FlexLine) {
|
|
// Omit divider
|
|
this._flexLines.push(flexLine);
|
|
}
|
|
|
|
private _determineMainSize(flexDirection: FlexDirection, widthMeasureSpec: number, heightMeasureSpec: number) {
|
|
let mainSize: number;
|
|
let paddingAlongMainAxis: number;
|
|
|
|
switch (flexDirection) {
|
|
case FlexDirection.ROW:
|
|
case FlexDirection.ROW_REVERSE:
|
|
let widthMode = getMeasureSpecMode(widthMeasureSpec);
|
|
let widthSize = getMeasureSpecSize(widthMeasureSpec);
|
|
if (widthMode === EXACTLY) {
|
|
mainSize = widthSize;
|
|
} else {
|
|
mainSize = this._getLargestMainSize();
|
|
}
|
|
paddingAlongMainAxis = this.effectivePaddingLeft + this.effectivePaddingRight;
|
|
break;
|
|
case FlexDirection.COLUMN:
|
|
case FlexDirection.COLUMN_REVERSE:
|
|
let heightMode = getMeasureSpecMode(heightMeasureSpec);
|
|
let heightSize = getMeasureSpecSize(heightMeasureSpec);
|
|
if (heightMode === EXACTLY) {
|
|
mainSize = heightSize;
|
|
} else {
|
|
mainSize = this._getLargestMainSize();
|
|
}
|
|
paddingAlongMainAxis = this.effectivePaddingTop + this.effectivePaddingBottom;
|
|
break;
|
|
default:
|
|
throw new Error("Invalid flex direction: " + flexDirection);
|
|
}
|
|
|
|
let childIndex = 0;
|
|
this._flexLines.forEach(flexLine => {
|
|
if (flexLine.mainSize < mainSize) {
|
|
childIndex = this._expandFlexItems(flexLine, flexDirection, mainSize, paddingAlongMainAxis, childIndex);
|
|
} else {
|
|
childIndex = this._shrinkFlexItems(flexLine, flexDirection, mainSize, paddingAlongMainAxis, childIndex);
|
|
}
|
|
});
|
|
}
|
|
|
|
private _expandFlexItems(flexLine: FlexLine, flexDirection: FlexDirection, maxMainSize: number, paddingAlongMainAxis: number, startIndex: number) {
|
|
let childIndex = startIndex;
|
|
if (flexLine._totalFlexGrow <= 0 || maxMainSize < flexLine._mainSize) {
|
|
childIndex += flexLine._itemCount;
|
|
|
|
return childIndex;
|
|
}
|
|
let sizeBeforeExpand = flexLine._mainSize;
|
|
let needsReexpand = false;
|
|
let pendingSpace = maxMainSize - flexLine._mainSize;
|
|
let unitSpace = pendingSpace / flexLine._totalFlexGrow;
|
|
flexLine._mainSize = paddingAlongMainAxis + flexLine._dividerLengthInMainSize;
|
|
let accumulatedRoundError = 0;
|
|
for (let i = 0; i < flexLine.itemCount; i++) {
|
|
let child = this._getReorderedChildAt(childIndex);
|
|
if (child === null) {
|
|
continue;
|
|
} else if (child.isCollapsed) {
|
|
childIndex++;
|
|
continue;
|
|
}
|
|
const lp = child; // .style;
|
|
if (this._isMainAxisDirectionHorizontal(flexDirection)) {
|
|
if (!this._childrenFrozen[childIndex]) {
|
|
let flexGrow = FlexboxLayout.getFlexGrow(child);
|
|
let rawCalculatedWidth = child.getMeasuredWidth() + unitSpace * flexGrow + accumulatedRoundError;
|
|
let roundedCalculatedWidth = Math.round(rawCalculatedWidth);
|
|
// TODO: MAX_SIZE is so big, this is always false:
|
|
if (roundedCalculatedWidth > MAX_SIZE /* lp.maxWidth */) {
|
|
needsReexpand = true;
|
|
roundedCalculatedWidth = MAX_SIZE /* lp.maxWidth */;
|
|
this._childrenFrozen[childIndex] = true;
|
|
flexLine._totalFlexGrow -= flexGrow;
|
|
} else {
|
|
accumulatedRoundError = rawCalculatedWidth - roundedCalculatedWidth;
|
|
}
|
|
child.measure(makeMeasureSpec(roundedCalculatedWidth, EXACTLY), makeMeasureSpec(child.getMeasuredHeight(), EXACTLY));
|
|
}
|
|
flexLine._mainSize += child.getMeasuredWidth() + lp.effectiveMarginLeft + lp.effectiveMarginRight;
|
|
} else {
|
|
if (!this._childrenFrozen[childIndex]) {
|
|
let flexGrow = FlexboxLayout.getFlexGrow(child);
|
|
let rawCalculatedHeight = child.getMeasuredHeight() + unitSpace * flexGrow + accumulatedRoundError;
|
|
let roundedCalculatedHeight = Math.round(rawCalculatedHeight);
|
|
// TODO: MAX_SIZE is so big this is always false:
|
|
if (roundedCalculatedHeight > MAX_SIZE /*lp.maxHeight*/) {
|
|
needsReexpand = true;
|
|
roundedCalculatedHeight = MAX_SIZE /*lp.maxHeight*/;
|
|
this._childrenFrozen[childIndex] = true;
|
|
flexLine._totalFlexGrow -= flexGrow;
|
|
} else {
|
|
accumulatedRoundError = rawCalculatedHeight - roundedCalculatedHeight;
|
|
}
|
|
child.measure(makeMeasureSpec(child.getMeasuredWidth(), EXACTLY), makeMeasureSpec(roundedCalculatedHeight, EXACTLY));
|
|
}
|
|
flexLine._mainSize += child.getMeasuredHeight() + lp.effectiveMarginTop + lp.effectiveMarginBottom;
|
|
}
|
|
childIndex++;
|
|
}
|
|
|
|
if (needsReexpand && sizeBeforeExpand !== flexLine._mainSize) {
|
|
this._expandFlexItems(flexLine, flexDirection, maxMainSize, paddingAlongMainAxis, startIndex);
|
|
}
|
|
|
|
return childIndex;
|
|
}
|
|
|
|
private _shrinkFlexItems(flexLine: FlexLine, flexDirection: FlexDirection, maxMainSize: number, paddingAlongMainAxis: number, startIndex: number): number {
|
|
let childIndex = startIndex;
|
|
let sizeBeforeShrink = flexLine._mainSize;
|
|
if (flexLine._totalFlexShrink <= 0 || maxMainSize > flexLine._mainSize) {
|
|
childIndex += flexLine.itemCount;
|
|
|
|
return childIndex;
|
|
}
|
|
let needsReshrink = false;
|
|
let unitShrink = (flexLine._mainSize - maxMainSize) / flexLine._totalFlexShrink;
|
|
let accumulatedRoundError = 0;
|
|
flexLine._mainSize = paddingAlongMainAxis + flexLine._dividerLengthInMainSize;
|
|
for (let i = 0; i < flexLine.itemCount; i++) {
|
|
let child = this._getReorderedChildAt(childIndex);
|
|
if (child === null) {
|
|
continue;
|
|
} else if (child.isCollapsed) {
|
|
childIndex++;
|
|
continue;
|
|
}
|
|
const lp = child; // .style;
|
|
if (this._isMainAxisDirectionHorizontal(flexDirection)) {
|
|
// The direction of main axis is horizontal
|
|
if (!this._childrenFrozen[childIndex]) {
|
|
let flexShrink = FlexboxLayout.getFlexShrink(child);
|
|
let rawCalculatedWidth = child.getMeasuredWidth() - unitShrink * flexShrink + accumulatedRoundError;
|
|
let roundedCalculatedWidth = Math.round(rawCalculatedWidth);
|
|
let minWidth = child.effectiveMinWidth;
|
|
child.effectiveMinWidth = 0;
|
|
if (roundedCalculatedWidth < minWidth) {
|
|
needsReshrink = true;
|
|
roundedCalculatedWidth = minWidth;
|
|
this._childrenFrozen[childIndex] = true;
|
|
flexLine._totalFlexShrink -= flexShrink;
|
|
} else {
|
|
accumulatedRoundError = rawCalculatedWidth - roundedCalculatedWidth;
|
|
}
|
|
|
|
const childWidthMeasureSpec = makeMeasureSpec(roundedCalculatedWidth, EXACTLY);
|
|
|
|
// NOTE: for controls that support internal content wrapping (e.g. UILabel) reducing the width
|
|
// might result in increased height e.g. text that could be shown on one line for larger
|
|
// width needs to be wrapped in two when width is reduced.
|
|
// As a result we cannot unconditionally measure with EXACTLY the current measured height
|
|
const childHeightMeasureSpec = FlexboxLayout.getChildMeasureSpec(this._currentHeightMeasureSpec,
|
|
lp.effectivePaddingTop + lp.effectivePaddingBottom + lp.effectiveMarginTop
|
|
+ lp.effectiveMarginBottom, lp.effectiveHeight < 0 ? WRAP_CONTENT : lp.effectiveHeight);
|
|
|
|
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
|
|
child.effectiveMinWidth = minWidth;
|
|
|
|
// make sure crossSize is up-to-date as child calculated height might have increased
|
|
flexLine._crossSize = Math.max(
|
|
flexLine._crossSize,
|
|
child.getMeasuredHeight() + lp.effectiveMarginTop + lp.effectiveMarginBottom
|
|
);
|
|
}
|
|
flexLine._mainSize += child.getMeasuredWidth() + lp.effectiveMarginLeft + lp.effectiveMarginRight;
|
|
} else {
|
|
if (!this._childrenFrozen[childIndex]) {
|
|
let flexShrink = FlexboxLayout.getFlexShrink(child);
|
|
let rawCalculatedHeight = child.getMeasuredHeight() - unitShrink * flexShrink + accumulatedRoundError;
|
|
let roundedCalculatedHeight = Math.round(rawCalculatedHeight);
|
|
const minHeight = child.effectiveMinHeight;
|
|
child.effectiveMinHeight = 0;
|
|
if (roundedCalculatedHeight < minHeight) {
|
|
needsReshrink = true;
|
|
roundedCalculatedHeight = minHeight;
|
|
this._childrenFrozen[childIndex] = true;
|
|
flexLine._totalFlexShrink -= flexShrink;
|
|
} else {
|
|
accumulatedRoundError = rawCalculatedHeight - roundedCalculatedHeight;
|
|
}
|
|
child.measure(makeMeasureSpec(child.getMeasuredWidth(), EXACTLY), makeMeasureSpec(roundedCalculatedHeight, EXACTLY));
|
|
child.effectiveMinHeight = minHeight;
|
|
}
|
|
flexLine._mainSize += child.getMeasuredHeight() + lp.effectiveMarginTop + lp.effectiveMarginBottom;
|
|
}
|
|
childIndex++;
|
|
}
|
|
|
|
if (needsReshrink && sizeBeforeShrink !== flexLine._mainSize) {
|
|
this._shrinkFlexItems(flexLine, flexDirection, maxMainSize, paddingAlongMainAxis, startIndex);
|
|
}
|
|
|
|
return childIndex;
|
|
}
|
|
|
|
private _determineCrossSize(flexDirection: FlexDirection, widthMeasureSpec: number, heightMeasureSpec: number, paddingAlongCrossAxis: number) {
|
|
let mode;
|
|
let size;
|
|
switch (flexDirection) {
|
|
case FlexDirection.ROW:
|
|
case FlexDirection.ROW_REVERSE:
|
|
mode = getMeasureSpecMode(heightMeasureSpec);
|
|
size = getMeasureSpecSize(heightMeasureSpec);
|
|
break;
|
|
case FlexDirection.COLUMN:
|
|
case FlexDirection.COLUMN_REVERSE:
|
|
mode = getMeasureSpecMode(widthMeasureSpec);
|
|
size = getMeasureSpecSize(widthMeasureSpec);
|
|
break;
|
|
default:
|
|
throw new Error("Invalid flex direction: " + flexDirection);
|
|
}
|
|
if (mode === EXACTLY) {
|
|
let totalCrossSize = this._getSumOfCrossSize() + paddingAlongCrossAxis;
|
|
if (this._flexLines.length === 1) {
|
|
this._flexLines[0]._crossSize = size - paddingAlongCrossAxis;
|
|
} else if (this._flexLines.length >= 2 && totalCrossSize < size) {
|
|
switch (this.alignContent) {
|
|
case AlignContent.STRETCH:
|
|
(() => {
|
|
let freeSpaceUnit = (size - totalCrossSize) / this._flexLines.length;
|
|
let accumulatedError = 0;
|
|
for (let i = 0, flexLinesSize = this._flexLines.length; i < flexLinesSize; i++) {
|
|
let flexLine = this._flexLines[i];
|
|
let newCrossSizeAsFloat = flexLine._crossSize + freeSpaceUnit;
|
|
if (i === this._flexLines.length - 1) {
|
|
newCrossSizeAsFloat += accumulatedError;
|
|
accumulatedError = 0;
|
|
}
|
|
let newCrossSize = Math.round(newCrossSizeAsFloat);
|
|
accumulatedError += (newCrossSizeAsFloat - newCrossSize);
|
|
if (accumulatedError > 1) {
|
|
newCrossSize += 1;
|
|
accumulatedError -= 1;
|
|
} else if (accumulatedError < -1) {
|
|
newCrossSize -= 1;
|
|
accumulatedError += 1;
|
|
}
|
|
flexLine._crossSize = newCrossSize;
|
|
}
|
|
})();
|
|
break;
|
|
case AlignContent.SPACE_AROUND:
|
|
(() => {
|
|
let spaceTopAndBottom = size - totalCrossSize;
|
|
let numberOfSpaces = this._flexLines.length * 2;
|
|
spaceTopAndBottom = spaceTopAndBottom / numberOfSpaces;
|
|
let newFlexLines: FlexLine[] = [];
|
|
let dummySpaceFlexLine = new FlexLine();
|
|
dummySpaceFlexLine._crossSize = spaceTopAndBottom;
|
|
this._flexLines.forEach(flexLine => {
|
|
newFlexLines.push(dummySpaceFlexLine);
|
|
newFlexLines.push(flexLine);
|
|
newFlexLines.push(dummySpaceFlexLine);
|
|
});
|
|
this._flexLines = newFlexLines;
|
|
})();
|
|
break;
|
|
case AlignContent.SPACE_BETWEEN:
|
|
(() => {
|
|
let spaceBetweenFlexLine = size - totalCrossSize;
|
|
let numberOfSpaces = this._flexLines.length - 1;
|
|
spaceBetweenFlexLine = spaceBetweenFlexLine / numberOfSpaces;
|
|
let accumulatedError = 0;
|
|
let newFlexLines: FlexLine[] = [];
|
|
for (let i = 0, flexLineSize = this._flexLines.length; i < flexLineSize; i++) {
|
|
let flexLine = this._flexLines[i];
|
|
newFlexLines.push(flexLine);
|
|
|
|
if (i !== this._flexLines.length - 1) {
|
|
let dummySpaceFlexLine = new FlexLine();
|
|
if (i === this._flexLines.length - 2) {
|
|
dummySpaceFlexLine._crossSize = Math.round(spaceBetweenFlexLine + accumulatedError);
|
|
accumulatedError = 0;
|
|
} else {
|
|
dummySpaceFlexLine._crossSize = Math.round(spaceBetweenFlexLine);
|
|
}
|
|
accumulatedError += (spaceBetweenFlexLine - dummySpaceFlexLine._crossSize);
|
|
if (accumulatedError > 1) {
|
|
dummySpaceFlexLine._crossSize += 1;
|
|
accumulatedError -= 1;
|
|
} else if (accumulatedError < -1) {
|
|
dummySpaceFlexLine._crossSize -= 1;
|
|
accumulatedError += 1;
|
|
}
|
|
newFlexLines.push(dummySpaceFlexLine);
|
|
}
|
|
}
|
|
this._flexLines = newFlexLines;
|
|
})();
|
|
break;
|
|
case AlignContent.CENTER: {
|
|
let spaceAboveAndBottom = size - totalCrossSize;
|
|
spaceAboveAndBottom = spaceAboveAndBottom / 2;
|
|
let newFlexLines: FlexLine[] = [];
|
|
let dummySpaceFlexLine = new FlexLine();
|
|
dummySpaceFlexLine._crossSize = spaceAboveAndBottom;
|
|
for (let i = 0, flexLineSize = this._flexLines.length; i < flexLineSize; i++) {
|
|
if (i === 0) {
|
|
newFlexLines.push(dummySpaceFlexLine);
|
|
}
|
|
let flexLine = this._flexLines[i];
|
|
newFlexLines.push(flexLine);
|
|
if (i === this._flexLines.length - 1) {
|
|
newFlexLines.push(dummySpaceFlexLine);
|
|
}
|
|
}
|
|
this._flexLines = newFlexLines;
|
|
break;
|
|
}
|
|
case AlignContent.FLEX_END: {
|
|
let spaceTop = size - totalCrossSize;
|
|
let dummySpaceFlexLine = new FlexLine();
|
|
dummySpaceFlexLine._crossSize = spaceTop;
|
|
this._flexLines.unshift(dummySpaceFlexLine);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private _stretchViews(flexDirection: FlexDirection, alignItems: AlignItems) {
|
|
if (alignItems === AlignItems.STRETCH) {
|
|
let viewIndex = 0;
|
|
this._flexLines.forEach(flexLine => {
|
|
for (let i = 0; i < flexLine.itemCount; i++ , viewIndex++) {
|
|
let view = this._getReorderedChildAt(viewIndex);
|
|
let alignSelf = FlexboxLayout.getAlignSelf(view);
|
|
if (alignSelf !== "auto" && alignSelf !== "stretch") {
|
|
continue;
|
|
}
|
|
switch (flexDirection) {
|
|
case FlexDirection.ROW:
|
|
case FlexDirection.ROW_REVERSE:
|
|
this._stretchViewVertically(view, flexLine._crossSize);
|
|
break;
|
|
case FlexDirection.COLUMN:
|
|
case FlexDirection.COLUMN_REVERSE:
|
|
this._stretchViewHorizontally(view, flexLine._crossSize);
|
|
break;
|
|
default:
|
|
throw new Error("Invalid flex direction: " + flexDirection);
|
|
}
|
|
}
|
|
});
|
|
} else {
|
|
this._flexLines.forEach(flexLine => {
|
|
flexLine._indicesAlignSelfStretch.forEach(index => {
|
|
let view = this._getReorderedChildAt(index);
|
|
switch (flexDirection) {
|
|
case FlexDirection.ROW:
|
|
case FlexDirection.ROW_REVERSE:
|
|
this._stretchViewVertically(view, flexLine._crossSize);
|
|
break;
|
|
case FlexDirection.COLUMN:
|
|
case FlexDirection.COLUMN_REVERSE:
|
|
this._stretchViewHorizontally(view, flexLine._crossSize);
|
|
break;
|
|
default:
|
|
throw new Error("Invalid flex direction: " + flexDirection);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
private _stretchViewVertically(view: View, crossSize: number) {
|
|
let newHeight = crossSize - view.effectiveMarginTop - view.effectiveMarginBottom;
|
|
newHeight = Math.max(newHeight, 0);
|
|
let originalMeasuredWidth = view.getMeasuredWidth();
|
|
let childWidthMeasureSpec = FlexboxLayout.getChildMeasureSpec(this._currentWidthMeasureSpec,
|
|
view.effectivePaddingLeft + view.effectivePaddingRight + view.effectiveMarginLeft
|
|
+ view.effectiveMarginRight, view.effectiveWidth < 0 ? WRAP_CONTENT : Math.min(view.effectiveWidth, originalMeasuredWidth));
|
|
|
|
view.measure(childWidthMeasureSpec, makeMeasureSpec(newHeight, EXACTLY));
|
|
|
|
if (originalMeasuredWidth > view.getMeasuredWidth()) {
|
|
childWidthMeasureSpec = makeMeasureSpec(originalMeasuredWidth, EXACTLY);
|
|
view.measure(childWidthMeasureSpec, makeMeasureSpec(newHeight, EXACTLY));
|
|
}
|
|
}
|
|
|
|
private _stretchViewHorizontally(view: View, crossSize: number) {
|
|
let newWidth = crossSize - view.effectiveMarginLeft - view.effectiveMarginRight;
|
|
newWidth = Math.max(newWidth, 0);
|
|
view.measure(makeMeasureSpec(newWidth, EXACTLY), makeMeasureSpec(view.getMeasuredHeight(), EXACTLY));
|
|
}
|
|
|
|
private _setMeasuredDimensionForFlex(flexDirection: FlexDirection, widthMeasureSpec: number, heightMeasureSpec: number, childState: number) {
|
|
let widthMode = getMeasureSpecMode(widthMeasureSpec);
|
|
let widthSize = getMeasureSpecSize(widthMeasureSpec);
|
|
let heightMode = getMeasureSpecMode(heightMeasureSpec);
|
|
let heightSize = getMeasureSpecSize(heightMeasureSpec);
|
|
let calculatedMaxHeight;
|
|
let calculatedMaxWidth;
|
|
switch (flexDirection) {
|
|
case FlexDirection.ROW:
|
|
case FlexDirection.ROW_REVERSE:
|
|
calculatedMaxHeight = this._getSumOfCrossSize() + this.effectivePaddingTop + this.effectivePaddingBottom;
|
|
calculatedMaxWidth = this._getLargestMainSize();
|
|
break;
|
|
case FlexDirection.COLUMN:
|
|
case FlexDirection.COLUMN_REVERSE:
|
|
calculatedMaxHeight = this._getLargestMainSize();
|
|
calculatedMaxWidth = this._getSumOfCrossSize() + this.effectivePaddingLeft + this.effectivePaddingRight;
|
|
break;
|
|
default:
|
|
throw new Error("Invalid flex direction: " + flexDirection);
|
|
}
|
|
|
|
let widthSizeAndState;
|
|
switch (widthMode) {
|
|
case EXACTLY:
|
|
if (widthSize < calculatedMaxWidth) {
|
|
childState = View.combineMeasuredStates(childState, MEASURED_STATE_TOO_SMALL);
|
|
}
|
|
widthSizeAndState = View.resolveSizeAndState(widthSize, widthSize, widthMode, childState);
|
|
break;
|
|
case AT_MOST: {
|
|
if (widthSize < calculatedMaxWidth) {
|
|
childState = View.combineMeasuredStates(childState, MEASURED_STATE_TOO_SMALL);
|
|
} else {
|
|
widthSize = calculatedMaxWidth;
|
|
}
|
|
widthSizeAndState = View.resolveSizeAndState(widthSize, widthSize, widthMode, childState);
|
|
break;
|
|
}
|
|
case UNSPECIFIED: {
|
|
widthSizeAndState = View.resolveSizeAndState(calculatedMaxWidth, widthSize, widthMode, childState);
|
|
break;
|
|
}
|
|
default:
|
|
throw new Error("Unknown width mode is set: " + widthMode);
|
|
}
|
|
let heightSizeAndState;
|
|
switch (heightMode) {
|
|
case EXACTLY:
|
|
if (heightSize < calculatedMaxHeight) {
|
|
childState = View.combineMeasuredStates(childState, MEASURED_STATE_TOO_SMALL >> layout.MEASURED_HEIGHT_STATE_SHIFT);
|
|
}
|
|
heightSizeAndState = View.resolveSizeAndState(heightSize, heightSize, heightMode, childState);
|
|
break;
|
|
case AT_MOST: {
|
|
if (heightSize < calculatedMaxHeight) {
|
|
childState = View.combineMeasuredStates(childState, MEASURED_STATE_TOO_SMALL >> layout.MEASURED_HEIGHT_STATE_SHIFT);
|
|
} else {
|
|
heightSize = calculatedMaxHeight;
|
|
}
|
|
|
|
heightSizeAndState = View.resolveSizeAndState(heightSize, heightSize, heightMode, childState);
|
|
break;
|
|
}
|
|
case UNSPECIFIED: {
|
|
heightSizeAndState = View.resolveSizeAndState(calculatedMaxHeight, heightSize, heightMode, childState);
|
|
break;
|
|
}
|
|
default:
|
|
throw new Error("Unknown height mode is set: " + heightMode);
|
|
}
|
|
this.setMeasuredDimension(widthSizeAndState, heightSizeAndState);
|
|
}
|
|
|
|
private _isWrapRequired(child: View, mode: number, maxSize: number, currentLength: number, childLength: number, childAbsoluteIndex: number, childRelativeIndexInFlexLine: number): boolean {
|
|
if (this.flexWrap === FlexWrap.NOWRAP) {
|
|
return false;
|
|
}
|
|
if (FlexboxLayout.getFlexWrapBefore(child)) {
|
|
return true;
|
|
}
|
|
if (mode === UNSPECIFIED) {
|
|
return false;
|
|
}
|
|
|
|
// Omit divider
|
|
|
|
return maxSize < currentLength + childLength;
|
|
}
|
|
|
|
private _getLargestMainSize(): number {
|
|
return this._flexLines.reduce((max, flexLine) => Math.max(max, flexLine.mainSize), Number.MIN_VALUE);
|
|
}
|
|
|
|
private _getSumOfCrossSize(): number {
|
|
// Omit divider
|
|
return this._flexLines.reduce((sum, flexLine) => sum + flexLine._crossSize, 0);
|
|
}
|
|
|
|
private _isMainAxisDirectionHorizontal(flexDirection: FlexDirection): boolean {
|
|
return flexDirection === FlexDirection.ROW || flexDirection === FlexDirection.ROW_REVERSE;
|
|
}
|
|
|
|
public onLayout(left: number, top: number, right: number, bottom: number) {
|
|
const insets = this.getSafeAreaInsets();
|
|
|
|
let isRtl;
|
|
switch (this.flexDirection) {
|
|
case FlexDirection.ROW:
|
|
isRtl = false;
|
|
this._layoutHorizontal(isRtl, left, top, right, bottom, insets);
|
|
break;
|
|
case FlexDirection.ROW_REVERSE:
|
|
isRtl = true;
|
|
this._layoutHorizontal(isRtl, left, top, right, bottom, insets);
|
|
break;
|
|
case FlexDirection.COLUMN:
|
|
isRtl = false;
|
|
if (this.flexWrap === FlexWrap.WRAP_REVERSE) {
|
|
isRtl = !isRtl;
|
|
}
|
|
this._layoutVertical(isRtl, false, left, top, right, bottom, insets);
|
|
break;
|
|
case FlexDirection.COLUMN_REVERSE:
|
|
isRtl = false;
|
|
if (this.flexWrap === FlexWrap.WRAP_REVERSE) {
|
|
isRtl = !isRtl;
|
|
}
|
|
this._layoutVertical(isRtl, true, left, top, right, bottom, insets);
|
|
break;
|
|
default:
|
|
throw new Error("Invalid flex direction is set: " + this.flexDirection);
|
|
}
|
|
}
|
|
|
|
private _layoutHorizontal(isRtl: boolean, left: number, top: number, right: number, bottom: number, insets: { left, top, right, bottom }) {
|
|
// include insets
|
|
let paddingLeft = this.effectivePaddingLeft + insets.left;
|
|
let paddingTop = this.effectivePaddingTop + insets.top;
|
|
let paddingRight = this.effectivePaddingRight + insets.right;
|
|
let paddingBottom = this.effectivePaddingBottom + insets.bottom;
|
|
|
|
let childLeft;
|
|
let currentViewIndex = 0;
|
|
|
|
let height = bottom - top;
|
|
let width = right - left;
|
|
|
|
// include insets
|
|
let childBottom = height - paddingBottom;
|
|
let childTop = paddingTop;
|
|
|
|
let childRight;
|
|
this._flexLines.forEach((flexLine, i) => {
|
|
|
|
// Omit divider
|
|
|
|
let spaceBetweenItem = 0.0;
|
|
switch (this.justifyContent) {
|
|
case JustifyContent.FLEX_START:
|
|
childLeft = paddingLeft;
|
|
childRight = width - paddingRight;
|
|
break;
|
|
case JustifyContent.FLEX_END:
|
|
childLeft = width - flexLine._mainSize + paddingRight;
|
|
childRight = flexLine._mainSize - paddingLeft;
|
|
break;
|
|
case JustifyContent.CENTER:
|
|
childLeft = paddingLeft + (width - insets.left - insets.right - flexLine._mainSize) / 2.0;
|
|
childRight = width - paddingRight - (width - insets.left - insets.right - flexLine._mainSize) / 2.0;
|
|
break;
|
|
case JustifyContent.SPACE_AROUND:
|
|
if (flexLine._itemCount !== 0) {
|
|
spaceBetweenItem = (width - insets.left - insets.right - flexLine.mainSize) / flexLine._itemCount;
|
|
}
|
|
childLeft = paddingLeft + spaceBetweenItem / 2.0;
|
|
childRight = width - paddingRight - spaceBetweenItem / 2.0;
|
|
break;
|
|
case JustifyContent.SPACE_BETWEEN:
|
|
childLeft = paddingLeft;
|
|
let denominator = flexLine.itemCount !== 1 ? flexLine.itemCount - 1 : 1.0;
|
|
spaceBetweenItem = (width - insets.left - insets.right - flexLine.mainSize) / denominator;
|
|
childRight = width - paddingRight;
|
|
break;
|
|
default:
|
|
throw new Error("Invalid justifyContent is set: " + this.justifyContent);
|
|
}
|
|
spaceBetweenItem = Math.max(spaceBetweenItem, 0);
|
|
|
|
for (let j = 0; j < flexLine.itemCount; j++) {
|
|
let child = this._getReorderedChildAt(currentViewIndex);
|
|
if (child === null) {
|
|
continue;
|
|
} else if (child.isCollapsed) {
|
|
currentViewIndex++;
|
|
continue;
|
|
}
|
|
const lp = child; // .style;
|
|
childLeft += lp.effectiveMarginLeft;
|
|
childRight -= lp.effectiveMarginRight;
|
|
|
|
// Omit divider
|
|
|
|
if (this.flexWrap === FlexWrap.WRAP_REVERSE) {
|
|
if (isRtl) {
|
|
this._layoutSingleChildHorizontal(child, flexLine, this.flexWrap, this.alignItems,
|
|
Math.round(childRight) - child.getMeasuredWidth(),
|
|
childBottom - child.getMeasuredHeight(), Math.round(childRight),
|
|
childBottom);
|
|
} else {
|
|
this._layoutSingleChildHorizontal(child, flexLine, this.flexWrap, this.alignItems,
|
|
Math.round(childLeft), childBottom - child.getMeasuredHeight(),
|
|
Math.round(childLeft) + child.getMeasuredWidth(),
|
|
childBottom);
|
|
}
|
|
} else {
|
|
if (isRtl) {
|
|
this._layoutSingleChildHorizontal(child, flexLine, this.flexWrap, this.alignItems,
|
|
Math.round(childRight) - child.getMeasuredWidth(), childTop,
|
|
Math.round(childRight), childTop + child.getMeasuredHeight());
|
|
} else {
|
|
this._layoutSingleChildHorizontal(child, flexLine, this.flexWrap, this.alignItems,
|
|
Math.round(childLeft), childTop,
|
|
Math.round(childLeft) + child.getMeasuredWidth(),
|
|
childTop + child.getMeasuredHeight());
|
|
}
|
|
}
|
|
childLeft += child.getMeasuredWidth() + spaceBetweenItem + lp.effectiveMarginRight;
|
|
childRight -= child.getMeasuredWidth() + spaceBetweenItem + lp.effectiveMarginLeft;
|
|
currentViewIndex++;
|
|
|
|
let bounds = child._getCurrentLayoutBounds();
|
|
flexLine._left = Math.min(flexLine._left, bounds.left - lp.effectiveMarginLeft);
|
|
flexLine._top = Math.min(flexLine._top, bounds.top - lp.effectiveMarginTop);
|
|
flexLine._right = Math.max(flexLine._right, bounds.right + lp.effectiveMarginRight);
|
|
flexLine._bottom = Math.max(flexLine._bottom, bounds.bottom + lp.effectiveMarginBottom);
|
|
}
|
|
|
|
childTop += flexLine._crossSize;
|
|
childBottom -= flexLine._crossSize;
|
|
});
|
|
}
|
|
|
|
private _layoutSingleChildHorizontal(view: View, flexLine: FlexLine, flexWrap: FlexWrap, alignItems: AlignItems, left: number, top: number, right: number, bottom: number): void {
|
|
let lp = view; // .style;
|
|
|
|
let alignSelf = FlexboxLayout.getAlignSelf(view);
|
|
if (alignSelf !== "auto") {
|
|
alignItems = alignSelf;
|
|
}
|
|
|
|
let crossSize = flexLine._crossSize;
|
|
switch (alignItems) {
|
|
case AlignItems.FLEX_START:
|
|
case AlignItems.STRETCH:
|
|
if (flexWrap !== FlexWrap.WRAP_REVERSE) {
|
|
view.layout(left, top + lp.effectiveMarginTop, right, bottom + lp.effectiveMarginTop);
|
|
} else {
|
|
view.layout(left, top - lp.effectiveMarginBottom, right, bottom - lp.effectiveMarginBottom);
|
|
}
|
|
break;
|
|
case AlignItems.BASELINE:
|
|
if (flexWrap !== FlexWrap.WRAP_REVERSE) {
|
|
let marginTop = flexLine._maxBaseline - FlexboxLayout.getBaseline(view);
|
|
marginTop = Math.max(marginTop, lp.effectiveMarginTop);
|
|
view.layout(left, top + marginTop, right, bottom + marginTop);
|
|
} else {
|
|
let marginBottom = flexLine._maxBaseline - view.getMeasuredHeight() + FlexboxLayout.getBaseline(view);
|
|
marginBottom = Math.max(marginBottom, lp.effectiveMarginBottom);
|
|
view.layout(left, top - marginBottom, right, bottom - marginBottom);
|
|
}
|
|
break;
|
|
case AlignItems.FLEX_END:
|
|
if (flexWrap !== FlexWrap.WRAP_REVERSE) {
|
|
view.layout(left,
|
|
top + crossSize - view.getMeasuredHeight() - lp.effectiveMarginBottom,
|
|
right, top + crossSize - lp.effectiveMarginBottom);
|
|
} else {
|
|
view.layout(left, top - crossSize + view.getMeasuredHeight() + lp.effectiveMarginTop,
|
|
right, bottom - crossSize + view.getMeasuredHeight() + lp.effectiveMarginTop);
|
|
}
|
|
break;
|
|
case AlignItems.CENTER:
|
|
let topFromCrossAxis = (crossSize - view.getMeasuredHeight()) / 2;
|
|
if (flexWrap !== FlexWrap.WRAP_REVERSE) {
|
|
view.layout(left, top + topFromCrossAxis + lp.effectiveMarginTop - lp.effectiveMarginBottom,
|
|
right, top + topFromCrossAxis + view.getMeasuredHeight() + lp.effectiveMarginTop
|
|
- lp.effectiveMarginBottom);
|
|
} else {
|
|
view.layout(left, top - topFromCrossAxis + lp.effectiveMarginTop - lp.effectiveMarginBottom,
|
|
right, top - topFromCrossAxis + view.getMeasuredHeight() + lp.effectiveMarginTop
|
|
- lp.effectiveMarginBottom);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
private _layoutVertical(isRtl: boolean, fromBottomToTop: boolean, left: number, top: number, right: number, bottom: number, insets: { left, top, right, bottom }) {
|
|
let paddingLeft = this.effectivePaddingLeft + insets.left;
|
|
let paddingTop = this.effectivePaddingTop + insets.top;
|
|
let paddingRight = this.effectivePaddingRight + insets.right;
|
|
let paddingBottom = this.effectivePaddingBottom + insets.bottom;
|
|
|
|
let childLeft = paddingLeft;
|
|
let currentViewIndex = 0;
|
|
|
|
let width = right - left;
|
|
let height = bottom - top;
|
|
let childRight = width - paddingRight;
|
|
|
|
let childTop;
|
|
let childBottom;
|
|
|
|
this._flexLines.forEach(flexLine => {
|
|
|
|
// Omit divider.
|
|
|
|
let spaceBetweenItem = 0.0;
|
|
|
|
switch (this.justifyContent) {
|
|
case JustifyContent.FLEX_START:
|
|
childTop = paddingTop;
|
|
childBottom = height - paddingBottom;
|
|
break;
|
|
case JustifyContent.FLEX_END:
|
|
childTop = height - flexLine._mainSize + paddingBottom;
|
|
childBottom = flexLine._mainSize - paddingTop;
|
|
break;
|
|
case JustifyContent.CENTER:
|
|
childTop = paddingTop + (height - insets.top - insets.bottom - flexLine._mainSize) / 2.0;
|
|
childBottom = height - paddingBottom - (height - insets.top - insets.bottom - flexLine._mainSize) / 2.0;
|
|
break;
|
|
case JustifyContent.SPACE_AROUND:
|
|
if (flexLine._itemCount !== 0) {
|
|
spaceBetweenItem = (height - insets.top - insets.bottom - flexLine._mainSize) / flexLine.itemCount;
|
|
}
|
|
childTop = paddingTop + spaceBetweenItem / 2.0;
|
|
childBottom = height - paddingBottom - spaceBetweenItem / 2.0;
|
|
break;
|
|
case JustifyContent.SPACE_BETWEEN:
|
|
childTop = paddingTop;
|
|
let denominator = flexLine.itemCount !== 1 ? flexLine.itemCount - 1 : 1.0;
|
|
spaceBetweenItem = (height - insets.top - insets.bottom - flexLine.mainSize) / denominator;
|
|
childBottom = height - paddingBottom;
|
|
break;
|
|
default:
|
|
throw new Error("Invalid justifyContent is set: " + this.justifyContent);
|
|
}
|
|
spaceBetweenItem = Math.max(spaceBetweenItem, 0);
|
|
|
|
for (let j = 0; j < flexLine.itemCount; j++) {
|
|
let child = this._getReorderedChildAt(currentViewIndex);
|
|
if (child === null) {
|
|
continue;
|
|
} else if (child.isCollapsed) {
|
|
currentViewIndex++;
|
|
continue;
|
|
}
|
|
const lp = child; // .style;
|
|
childTop += lp.effectiveMarginTop;
|
|
childBottom -= lp.effectiveMarginBottom;
|
|
|
|
// Omit divider.
|
|
|
|
if (isRtl) {
|
|
if (fromBottomToTop) {
|
|
this._layoutSingleChildVertical(child, flexLine, true, this.alignItems,
|
|
childRight - child.getMeasuredWidth(),
|
|
Math.round(childBottom) - child.getMeasuredHeight(), childRight,
|
|
Math.round(childBottom));
|
|
} else {
|
|
this._layoutSingleChildVertical(child, flexLine, true, this.alignItems,
|
|
childRight - child.getMeasuredWidth(), Math.round(childTop),
|
|
childRight, Math.round(childTop) + child.getMeasuredHeight());
|
|
}
|
|
} else {
|
|
if (fromBottomToTop) {
|
|
this._layoutSingleChildVertical(child, flexLine, false, this.alignItems,
|
|
childLeft, Math.round(childBottom) - child.getMeasuredHeight(),
|
|
childLeft + child.getMeasuredWidth(), Math.round(childBottom));
|
|
} else {
|
|
this._layoutSingleChildVertical(child, flexLine, false, this.alignItems,
|
|
childLeft, Math.round(childTop),
|
|
childLeft + child.getMeasuredWidth(),
|
|
Math.round(childTop) + child.getMeasuredHeight());
|
|
}
|
|
}
|
|
childTop += child.getMeasuredHeight() + spaceBetweenItem + lp.effectiveMarginBottom;
|
|
childBottom -= child.getMeasuredHeight() + spaceBetweenItem + lp.effectiveMarginTop;
|
|
currentViewIndex++;
|
|
|
|
let bounds = child._getCurrentLayoutBounds();
|
|
flexLine._left = Math.min(flexLine._left, bounds.left - lp.effectiveMarginLeft);
|
|
flexLine._top = Math.min(flexLine._top, bounds.top - lp.effectiveMarginTop);
|
|
flexLine._right = Math.max(flexLine._right, bounds.right + lp.effectiveMarginRight);
|
|
flexLine._bottom = Math.max(flexLine._bottom, bounds.bottom + lp.effectiveMarginBottom);
|
|
}
|
|
|
|
childLeft += flexLine.crossSize;
|
|
childRight -= flexLine.crossSize;
|
|
});
|
|
}
|
|
|
|
private _layoutSingleChildVertical(view: View, flexLine: FlexLine, isRtl: boolean, alignItems: AlignItems, left: number, top: number, right: number, bottom: number) {
|
|
let lp = view; // .style;
|
|
let alignSelf = FlexboxLayout.getAlignSelf(view);
|
|
if (alignSelf !== "auto") {
|
|
alignItems = alignSelf;
|
|
}
|
|
let crossSize = flexLine.crossSize;
|
|
switch (alignItems) {
|
|
case AlignItems.FLEX_START:
|
|
case AlignItems.STRETCH:
|
|
case AlignItems.BASELINE:
|
|
if (!isRtl) {
|
|
view.layout(left + lp.effectiveMarginLeft, top, right + lp.effectiveMarginLeft, bottom);
|
|
} else {
|
|
view.layout(left - lp.effectiveMarginRight, top, right - lp.effectiveMarginRight, bottom);
|
|
}
|
|
break;
|
|
case AlignItems.FLEX_END:
|
|
if (!isRtl) {
|
|
view.layout(left + crossSize - view.getMeasuredWidth() - lp.effectiveMarginRight,
|
|
top, right + crossSize - view.getMeasuredWidth() - lp.effectiveMarginRight,
|
|
bottom);
|
|
} else {
|
|
// If the flexWrap === FLEX_WRAP_WRAP_REVERSE, the direction of the
|
|
// flexEnd is flipped (from left to right).
|
|
view.layout(left - crossSize + view.getMeasuredWidth() + lp.effectiveMarginLeft, top,
|
|
right - crossSize + view.getMeasuredWidth() + lp.effectiveMarginLeft,
|
|
bottom);
|
|
}
|
|
break;
|
|
case AlignItems.CENTER:
|
|
let leftFromCrossAxis = (crossSize - view.getMeasuredWidth()) / 2;
|
|
if (!isRtl) {
|
|
view.layout(left + leftFromCrossAxis + lp.effectiveMarginLeft - lp.effectiveMarginRight,
|
|
top, right + leftFromCrossAxis + lp.effectiveMarginLeft - lp.effectiveMarginRight,
|
|
bottom);
|
|
} else {
|
|
view.layout(left - leftFromCrossAxis + lp.effectiveMarginLeft - lp.effectiveMarginRight,
|
|
top, right - leftFromCrossAxis + lp.effectiveMarginLeft - lp.effectiveMarginRight,
|
|
bottom);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Omit divider in onDraw(), drawDividersHorizontal, drawDividersVertical, drawVerticalDivider
|
|
|
|
// requestLayout on set flexDirection, set flexWrap, set justifyContent, set alignItems, set alignContent
|
|
|
|
// NOTE Consider moving to View if frequently used
|
|
private static getChildMeasureSpec(spec: number, padding: number, childDimension: number): number {
|
|
let specMode = layout.getMeasureSpecMode(spec);
|
|
let specSize = layout.getMeasureSpecSize(spec);
|
|
|
|
let size = Math.max(0, specSize - padding);
|
|
|
|
let resultSize = 0;
|
|
let resultMode = 0;
|
|
|
|
switch (specMode) {
|
|
// Parent has imposed an exact size on us
|
|
case EXACTLY:
|
|
if (childDimension >= 0) {
|
|
resultSize = childDimension;
|
|
resultMode = EXACTLY;
|
|
} else if (childDimension === MATCH_PARENT) {
|
|
resultSize = size;
|
|
resultMode = EXACTLY;
|
|
} else if (childDimension === WRAP_CONTENT) {
|
|
resultSize = size;
|
|
resultMode = AT_MOST;
|
|
}
|
|
break;
|
|
|
|
case AT_MOST:
|
|
if (childDimension >= 0) {
|
|
resultSize = childDimension;
|
|
resultMode = EXACTLY;
|
|
} else if (childDimension === MATCH_PARENT) {
|
|
resultSize = size;
|
|
resultMode = AT_MOST;
|
|
} else if (childDimension === WRAP_CONTENT) {
|
|
resultSize = size;
|
|
resultMode = AT_MOST;
|
|
}
|
|
break;
|
|
|
|
case UNSPECIFIED:
|
|
if (childDimension >= 0) {
|
|
resultSize = childDimension;
|
|
resultMode = EXACTLY;
|
|
} else if (childDimension === MATCH_PARENT) {
|
|
resultSize = View_sUseZeroUnspecifiedMeasureSpec ? 0 : size;
|
|
resultMode = UNSPECIFIED;
|
|
} else if (childDimension === WRAP_CONTENT) {
|
|
resultSize = View_sUseZeroUnspecifiedMeasureSpec ? 0 : size;
|
|
resultMode = UNSPECIFIED;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return layout.makeMeasureSpec(resultSize, resultMode);
|
|
}
|
|
}
|
|
|
|
export namespace FlexboxLayout {
|
|
export function getBaseline(child: View): number {
|
|
// TODO: Check if we support baseline for iOS.
|
|
return 0;
|
|
}
|
|
|
|
export function getPaddingStart(child: View): number {
|
|
return child.effectivePaddingLeft;
|
|
}
|
|
|
|
export function getPaddingEnd(child: View): number {
|
|
return child.effectivePaddingRight;
|
|
}
|
|
}
|