You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
netsurf/content/handlers/html/layout_flex.c

1120 lines
27 KiB

/*
* Copyright 2022 Michael Drake <tlsa@netsurf-browser.org>
*
* This file is part of NetSurf, http://www.netsurf-browser.org/
*
* NetSurf is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; version 2 of the License.
*
* NetSurf is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
/**
* \file
* HTML layout implementation: display: flex.
*
* Layout is carried out in two stages:
*
* 1. + calculation of minimum / maximum box widths, and
* + determination of whether block level boxes will have >zero height
*
* 2. + layout (position and dimensions)
*
* In most cases the functions for the two stages are a corresponding pair
* layout_minmax_X() and layout_X().
*/
#include <string.h>
#include "utils/log.h"
#include "utils/utils.h"
#include "html/box.h"
#include "html/html.h"
#include "html/private.h"
#include "html/box_inspect.h"
#include "html/layout_internal.h"
/**
* Flex item data
*/
struct flex_item_data {
enum css_flex_basis_e basis;
css_fixed basis_length;
css_unit basis_unit;
struct box *box;
css_fixed shrink;
css_fixed grow;
int min_main;
int max_main;
int min_cross;
int max_cross;
int target_main_size;
int base_size;
int main_size;
size_t line;
bool freeze;
bool min_violation;
bool max_violation;
};
/**
* Flex line data
*/
struct flex_line_data {
int main_size;
int cross_size;
int used_main_size;
int main_auto_margin_count;
int pos;
size_t first;
size_t count;
size_t frozen;
};
/**
* Flex layout context
*/
struct flex_ctx {
html_content *content;
const struct box *flex;
const css_unit_ctx *unit_len_ctx;
int main_size;
int cross_size;
int available_main;
int available_cross;
bool horizontal;
bool main_reversed;
enum css_flex_wrap_e wrap;
struct flex_items {
size_t count;
struct flex_item_data *data;
} item;
struct flex_lines {
size_t count;
size_t alloc;
struct flex_line_data *data;
} line;
};
/**
* Destroy a flex layout context
*
* \param[in] ctx Flex layout context
*/
static void layout_flex_ctx__destroy(struct flex_ctx *ctx)
{
if (ctx != NULL) {
free(ctx->item.data);
free(ctx->line.data);
free(ctx);
}
}
/**
* Create a flex layout context
*
* \param[in] content HTML content containing flex box
* \param[in] flex Box to create layout context for
* \return flex layout context or NULL on error
*/
static struct flex_ctx *layout_flex_ctx__create(
html_content *content,
const struct box *flex)
{
struct flex_ctx *ctx;
ctx = calloc(1, sizeof(*ctx));
if (ctx == NULL) {
return NULL;
}
ctx->line.alloc = 1;
ctx->item.count = box_count_children(flex);
ctx->item.data = calloc(ctx->item.count, sizeof(*ctx->item.data));
if (ctx->item.data == NULL) {
layout_flex_ctx__destroy(ctx);
return NULL;
}
ctx->line.alloc = 1;
ctx->line.data = calloc(ctx->line.alloc, sizeof(*ctx->line.data));
if (ctx->line.data == NULL) {
layout_flex_ctx__destroy(ctx);
return NULL;
}
ctx->flex = flex;
ctx->content = content;
ctx->unit_len_ctx = &content->unit_len_ctx;
ctx->wrap = css_computed_flex_wrap(flex->style);
ctx->horizontal = lh__flex_main_is_horizontal(flex);
ctx->main_reversed = lh__flex_direction_reversed(flex);
return ctx;
}
/**
* Find box side representing the start of flex container in main direction.
*
* \param[in] ctx Flex layout context.
* \return the start side.
*/
static enum box_side layout_flex__main_start_side(
const struct flex_ctx *ctx)
{
if (ctx->horizontal) {
return (ctx->main_reversed) ? RIGHT : LEFT;
} else {
return (ctx->main_reversed) ? BOTTOM : TOP;
}
}
/**
* Find box side representing the end of flex container in main direction.
*
* \param[in] ctx Flex layout context.
* \return the end side.
*/
static enum box_side layout_flex__main_end_side(
const struct flex_ctx *ctx)
{
if (ctx->horizontal) {
return (ctx->main_reversed) ? LEFT : RIGHT;
} else {
return (ctx->main_reversed) ? TOP : BOTTOM;
}
}
/**
* Perform layout on a flex item
*
* \param[in] ctx Flex layout context
* \param[in] item Item to lay out
* \param[in] available_width Available width for item in pixels
* \return true on success false on failure
*/
static bool layout_flex_item(
const struct flex_ctx *ctx,
const struct flex_item_data *item,
int available_width)
{
bool success;
struct box *b = item->box;
switch (b->type) {
case BOX_BLOCK:
success = layout_block_context(b, -1, ctx->content);
break;
case BOX_TABLE:
b->float_container = b->parent;
success = layout_table(b, available_width, ctx->content);
b->float_container = NULL;
break;
case BOX_FLEX:
b->float_container = b->parent;
success = layout_flex(b, available_width, ctx->content);
b->float_container = NULL;
break;
default:
assert(0 && "Bad flex item back type");
success = false;
break;
}
if (!success) {
NSLOG(flex, ERROR, "box %p: layout failed", b);
}
return success;
}
/**
* Calculate an item's base and target main sizes.
*
* \param[in] ctx Flex layout context
* \param[in] item Item to get sizes of
* \param[in] available_width Available width in pixels
* \return true on success false on failure
*/
static inline bool layout_flex__base_and_main_sizes(
const struct flex_ctx *ctx,
struct flex_item_data *item,
int available_width)
{
struct box *b = item->box;
int content_min_width = b->min_width;
int content_max_width = b->max_width;
int delta_outer_main = lh__delta_outer_main(ctx->flex, b);
NSLOG(flex, DEEPDEBUG, "box %p: delta_outer_main: %i",
b, delta_outer_main);
if (item->basis == CSS_FLEX_BASIS_SET) {
if (item->basis_unit == CSS_UNIT_PCT) {
item->base_size = FPCT_OF_INT_TOINT(
item->basis_length,
available_width);
} else {
item->base_size = FIXTOINT(css_unit_len2device_px(
b->style, ctx->unit_len_ctx,
item->basis_length,
item->basis_unit));
}
} else if (item->basis == CSS_FLEX_BASIS_AUTO) {
item->base_size = ctx->horizontal ? b->width : b->height;
} else {
item->base_size = AUTO;
}
if (ctx->horizontal == false) {
if (b->width == AUTO) {
b->width = min(max(content_min_width, available_width),
content_max_width);
b->width -= lh__delta_outer_width(b);
}
if (!layout_flex_item(ctx, item, b->width)) {
return false;
}
}
if (item->base_size == AUTO) {
if (ctx->horizontal == false) {
item->base_size = b->height;
} else {
item->base_size = content_max_width - delta_outer_main;
}
}
item->base_size += delta_outer_main;
if (ctx->horizontal) {
item->base_size = min(item->base_size, available_width);
item->base_size = max(item->base_size, content_min_width);
}
item->target_main_size = item->base_size;
item->main_size = item->base_size;
if (item->max_main > 0 &&
item->main_size > item->max_main + delta_outer_main) {
item->main_size = item->max_main + delta_outer_main;
}
if (item->main_size < item->min_main + delta_outer_main) {
item->main_size = item->min_main + delta_outer_main;
}
NSLOG(flex, DEEPDEBUG, "flex-item box: %p: base_size: %i, main_size %i",
b, item->base_size, item->main_size);
return true;
}
/**
* Fill out all item's data in a flex container.
*
* \param[in] ctx Flex layout context
* \param[in] flex Flex box
* \param[in] available_width Available width in pixels
*/
static void layout_flex_ctx__populate_item_data(
const struct flex_ctx *ctx,
const struct box *flex,
int available_width)
{
size_t i = 0;
bool horizontal = ctx->horizontal;
for (struct box *b = flex->children; b != NULL; b = b->next) {
struct flex_item_data *item = &ctx->item.data[i++];
b->float_container = b->parent;
layout_find_dimensions(ctx->unit_len_ctx, available_width, -1,
b, b->style, &b->width, &b->height,
horizontal ? &item->max_main : &item->max_cross,
horizontal ? &item->min_main : &item->min_cross,
horizontal ? &item->max_cross : &item->max_main,
horizontal ? &item->min_cross : &item->min_main,
b->margin, b->padding, b->border);
b->float_container = NULL;
NSLOG(flex, DEEPDEBUG, "flex-item box: %p: width: %i",
b, b->width);
item->box = b;
item->basis = css_computed_flex_basis(b->style,
&item->basis_length, &item->basis_unit);
css_computed_flex_shrink(b->style, &item->shrink);
css_computed_flex_grow(b->style, &item->grow);
layout_flex__base_and_main_sizes(ctx, item, available_width);
}
}
/**
* Ensure context's lines array has a free space
*
* \param[in] ctx Flex layout context
* \return true on success false on out of memory
*/
static bool layout_flex_ctx__ensure_line(struct flex_ctx *ctx)
{
struct flex_line_data *temp;
size_t line_alloc = ctx->line.alloc * 2;
if (ctx->line.alloc > ctx->line.count) {
return true;
}
temp = realloc(ctx->line.data, sizeof(*ctx->line.data) * line_alloc);
if (temp == NULL) {
return false;
}
ctx->line.data = temp;
memset(ctx->line.data + ctx->line.alloc, 0,
sizeof(*ctx->line.data) * (line_alloc - ctx->line.alloc));
ctx->line.alloc = line_alloc;
return true;
}
/**
* Assigns flex items to the line and returns the line
*
* \param[in] ctx Flex layout context
* \param[in] item_index Index to first item to assign to this line
* \return Pointer to the new line, or NULL on error.
*/
static struct flex_line_data *layout_flex__build_line(struct flex_ctx *ctx,
size_t item_index)
{
enum box_side start_side = layout_flex__main_start_side(ctx);
enum box_side end_side = layout_flex__main_end_side(ctx);
struct flex_line_data *line;
int used_main = 0;
if (!layout_flex_ctx__ensure_line(ctx)) {
return NULL;
}
line = &ctx->line.data[ctx->line.count];
line->first = item_index;
NSLOG(flex, DEEPDEBUG, "flex container %p: available main: %i",
ctx->flex, ctx->available_main);
while (item_index < ctx->item.count) {
struct flex_item_data *item = &ctx->item.data[item_index];
struct box *b = item->box;
int pos_main;
pos_main = ctx->horizontal ?
item->main_size :
b->height + lh__delta_outer_main(ctx->flex, b);
if (ctx->wrap == CSS_FLEX_WRAP_NOWRAP ||
pos_main + used_main <= ctx->available_main ||
lh__box_is_absolute(item->box) ||
ctx->available_main == AUTO ||
line->count == 0 ||
pos_main == 0) {
if (lh__box_is_absolute(item->box) == false) {
line->main_size += item->main_size;
used_main += pos_main;
if (b->margin[start_side] == AUTO) {
line->main_auto_margin_count++;
}
if (b->margin[end_side] == AUTO) {
line->main_auto_margin_count++;
}
}
item->line = ctx->line.count;
line->count++;
item_index++;
} else {
break;
}
}
if (line->count > 0) {
ctx->line.count++;
} else {
NSLOG(layout, ERROR, "Failed to fit any flex items");
}
return line;
}
/**
* Freeze an item on a line
*
* \param[in] line Line to containing item
* \param[in] item Item to freeze
*/
static inline void layout_flex__item_freeze(
struct flex_line_data *line,
struct flex_item_data *item)
{
item->freeze = true;
line->frozen++;
if (!lh__box_is_absolute(item->box)){
line->used_main_size += item->target_main_size;
}
NSLOG(flex, DEEPDEBUG, "flex-item box: %p: "
"Frozen at target_main_size: %i",
item->box, item->target_main_size);
}
/**
* Calculate remaining free space and unfrozen item factor sum
*
* \param[in] ctx Flex layout context
* \param[in] line Line to calculate free space on
* \param[out] unfrozen_factor_sum Returns sum of unfrozen item's flex factors
* \param[in] initial_free_main Initial free space in main direction
* \param[in] available_main Available space in main direction
* \param[in] grow Whether to grow or shrink
* return remaining free space on line
*/
static inline int layout_flex__remaining_free_main(
struct flex_ctx *ctx,
struct flex_line_data *line,
css_fixed *unfrozen_factor_sum,
int initial_free_main,
int available_main,
bool grow)
{
int remaining_free_main = available_main;
size_t item_count = line->first + line->count;
*unfrozen_factor_sum = 0;
for (size_t i = line->first; i < item_count; i++) {
struct flex_item_data *item = &ctx->item.data[i];
if (item->freeze) {
remaining_free_main -= item->target_main_size;
} else {
remaining_free_main -= item->base_size;
*unfrozen_factor_sum += grow ?
item->grow : item->shrink;
}
}
if (*unfrozen_factor_sum < F_1) {
int free_space = FIXTOINT(FMUL(INTTOFIX(initial_free_main),
*unfrozen_factor_sum));
if (free_space < remaining_free_main) {
remaining_free_main = free_space;
}
}
NSLOG(flex, DEEPDEBUG, "Remaining free space: %i",
remaining_free_main);
return remaining_free_main;
}
/**
* Clamp flex item target main size and get min/max violations
*
* \param[in] ctx Flex layout context
* \param[in] line Line to align items on
* return total violation in pixels
*/
static inline int layout_flex__get_min_max_violations(
struct flex_ctx *ctx,
struct flex_line_data *line)
{
int total_violation = 0;
size_t item_count = line->first + line->count;
for (size_t i = line->first; i < item_count; i++) {
struct flex_item_data *item = &ctx->item.data[i];
int target_main_size = item->target_main_size;
NSLOG(flex, DEEPDEBUG, "item %p: target_main_size: %i",
item->box, target_main_size);
if (item->freeze) {
continue;
}
if (item->max_main > 0 &&
target_main_size > item->max_main) {
target_main_size = item->max_main;
item->max_violation = true;
NSLOG(flex, DEEPDEBUG, "Violation: max_main: %i",
item->max_main);
}
if (target_main_size < item->min_main) {
target_main_size = item->min_main;
item->min_violation = true;
NSLOG(flex, DEEPDEBUG, "Violation: min_main: %i",
item->min_main);
}
if (target_main_size < item->box->min_width) {
target_main_size = item->box->min_width;
item->min_violation = true;
NSLOG(flex, DEEPDEBUG, "Violation: box min_width: %i",
item->box->min_width);
}
if (target_main_size < 0) {
target_main_size = 0;
item->min_violation = true;
NSLOG(flex, DEEPDEBUG, "Violation: less than 0");
}
total_violation += target_main_size - item->target_main_size;
item->target_main_size = target_main_size;
}
NSLOG(flex, DEEPDEBUG, "Total violation: %i", total_violation);
return total_violation;
}
/**
* Distribute remaining free space proportional to the flex factors.
*
* Remaining free space may be negative.
*
* \param[in] ctx Flex layout context
* \param[in] line Line to distribute free space on
* \param[in] unfrozen_factor_sum Sum of unfrozen item's flex factors
* \param[in] remaining_free_main Remaining free space in main direction
* \param[in] grow Whether to grow or shrink
*/
static inline void layout_flex__distribute_free_main(
struct flex_ctx *ctx,
struct flex_line_data *line,
css_fixed unfrozen_factor_sum,
int remaining_free_main,
bool grow)
{
size_t item_count = line->first + line->count;
if (grow) {
css_fixed remainder = 0;
for (size_t i = line->first; i < item_count; i++) {
struct flex_item_data *item = &ctx->item.data[i];
css_fixed result;
css_fixed ratio;
if (item->freeze) {
continue;
}
ratio = FDIV(item->grow, unfrozen_factor_sum);
result = FMUL(INTTOFIX(remaining_free_main), ratio) +
remainder;
item->target_main_size = item->base_size +
FIXTOINT(result);
remainder = FIXFRAC(result);
}
} else {
css_fixed scaled_shrink_factor_sum = 0;
css_fixed remainder = 0;
for (size_t i = line->first; i < item_count; i++) {
struct flex_item_data *item = &ctx->item.data[i];
css_fixed scaled_shrink_factor;
if (item->freeze) {
continue;
}
scaled_shrink_factor = FMUL(
item->shrink,
INTTOFIX(item->base_size));
scaled_shrink_factor_sum += scaled_shrink_factor;
}
for (size_t i = line->first; i < item_count; i++) {
struct flex_item_data *item = &ctx->item.data[i];
css_fixed scaled_shrink_factor;
css_fixed result;
css_fixed ratio;
if (item->freeze) {
continue;
} else if (scaled_shrink_factor_sum == 0) {
item->target_main_size = item->main_size;
layout_flex__item_freeze(line, item);
continue;
}
scaled_shrink_factor = FMUL(
item->shrink,
INTTOFIX(item->base_size));
ratio = FDIV(scaled_shrink_factor,
scaled_shrink_factor_sum);
result = FMUL(INTTOFIX(abs(remaining_free_main)),
ratio) + remainder;
item->target_main_size = item->base_size -
FIXTOINT(result);
remainder = FIXFRAC(result);
}
}
}
/**
* Resolve flexible item lengths along a line.
*
* See 9.7 of Tests CSS Flexible Box Layout Module Level 1.
*
* \param[in] ctx Flex layout context
* \param[in] line Line to resolve
* \return true on success, false on failure.
*/
static bool layout_flex__resolve_line(
struct flex_ctx *ctx,
struct flex_line_data *line)
{
size_t item_count = line->first + line->count;
int available_main = ctx->available_main;
int initial_free_main;
bool grow;
if (available_main == AUTO) {
available_main = INT_MAX;
}
grow = (line->main_size < available_main);
initial_free_main = available_main;
NSLOG(flex, DEEPDEBUG, "box %p: line %zu: first: %zu, count: %zu",
ctx->flex, line - ctx->line.data,
line->first, line->count);
NSLOG(flex, DEEPDEBUG, "Line main_size: %i, available_main: %i",
line->main_size, available_main);
for (size_t i = line->first; i < item_count; i++) {
struct flex_item_data *item = &ctx->item.data[i];
/* 3. Size inflexible items */
if (grow) {
if (item->grow == 0 ||
item->base_size > item->main_size) {
item->target_main_size = item->main_size;
layout_flex__item_freeze(line, item);
}
} else {
if (item->shrink == 0 ||
item->base_size < item->main_size) {
item->target_main_size = item->main_size;
layout_flex__item_freeze(line, item);
}
}
/* 4. Calculate initial free space */
if (item->freeze) {
initial_free_main -= item->target_main_size;
} else {
initial_free_main -= item->base_size;
}
}
/* 5. Loop */
while (line->frozen < line->count) {
css_fixed unfrozen_factor_sum;
int remaining_free_main;
int total_violation;
NSLOG(flex, DEEPDEBUG, "flex-container: %p: Resolver pass",
ctx->flex);
/* b */
remaining_free_main = layout_flex__remaining_free_main(ctx,
line, &unfrozen_factor_sum, initial_free_main,
available_main, grow);
/* c */
if (remaining_free_main != 0) {
layout_flex__distribute_free_main(ctx,
line, unfrozen_factor_sum,
remaining_free_main, grow);
}
/* d */
total_violation = layout_flex__get_min_max_violations(
ctx, line);
/* e */
for (size_t i = line->first; i < item_count; i++) {
struct flex_item_data *item = &ctx->item.data[i];
if (item->freeze) {
continue;
}
if (total_violation == 0 ||
(total_violation > 0 && item->min_violation) ||
(total_violation < 0 && item->max_violation)) {
layout_flex__item_freeze(line, item);
}
}
}
return true;
}
/**
* Position items along a line
*
* \param[in] ctx Flex layout context
* \param[in] line Line to resolve
* \return true on success, false on failure.
*/
static bool layout_flex__place_line_items_main(
struct flex_ctx *ctx,
struct flex_line_data *line)
{
int main_pos = ctx->flex->padding[layout_flex__main_start_side(ctx)];
int post_multiplier = ctx->main_reversed ? 0 : 1;
int pre_multiplier = ctx->main_reversed ? -1 : 0;
size_t item_count = line->first + line->count;
int extra_remainder = 0;
int extra = 0;
if (ctx->main_reversed) {
main_pos = lh__box_size_main(ctx->horizontal, ctx->flex) -
main_pos;
}
if (ctx->available_main != AUTO &&
ctx->available_main != UNKNOWN_WIDTH &&
ctx->available_main > line->used_main_size) {
if (line->main_auto_margin_count > 0) {
extra = ctx->available_main - line->used_main_size;
extra_remainder = extra % line->main_auto_margin_count;
extra /= line->main_auto_margin_count;
}
}
for (size_t i = line->first; i < item_count; i++) {
enum box_side main_end = ctx->horizontal ? RIGHT : BOTTOM;
enum box_side main_start = ctx->horizontal ? LEFT : TOP;
struct flex_item_data *item = &ctx->item.data[i];
struct box *b = item->box;
int extra_total = 0;
int extra_post = 0;
int extra_pre = 0;
int box_size_main;
int *box_pos_main;
if (ctx->horizontal) {
b->width = item->target_main_size -
lh__delta_outer_width(b);
if (!layout_flex_item(ctx, item, b->width)) {
return false;
}
}
box_size_main = lh__box_size_main(ctx->horizontal, b);
box_pos_main = ctx->horizontal ? &b->x : &b->y;
if (!lh__box_is_absolute(b)) {
if (b->margin[main_start] == AUTO) {
extra_pre = extra + extra_remainder;
}
if (b->margin[main_end] == AUTO) {
extra_post = extra + extra_remainder;
}
extra_total = extra_pre + extra_post;
main_pos += pre_multiplier *
(extra_total + box_size_main +
lh__delta_outer_main(ctx->flex, b));
}
*box_pos_main = main_pos + lh__non_auto_margin(b, main_start) +
extra_pre + b->border[main_start].width;
if (!lh__box_is_absolute(b)) {
int cross_size;
int box_size_cross = lh__box_size_cross(
ctx->horizontal, b);
main_pos += post_multiplier *
(extra_total + box_size_main +
lh__delta_outer_main(ctx->flex, b));
cross_size = box_size_cross + lh__delta_outer_cross(
ctx->flex, b);
if (line->cross_size < cross_size) {
line->cross_size = cross_size;
}
}
}
return true;
}
/**
* Collect items onto lines and place items along the lines
*
* \param[in] ctx Flex layout context
* \return true on success, false on failure.
*/
static bool layout_flex__collect_items_into_lines(
struct flex_ctx *ctx)
{
size_t pos = 0;
while (pos < ctx->item.count) {
struct flex_line_data *line;
line = layout_flex__build_line(ctx, pos);
if (line == NULL) {
return false;
}
pos += line->count;
NSLOG(flex, DEEPDEBUG, "flex-container: %p: "
"fitted: %zu (total: %zu/%zu)",
ctx->flex, line->count,
pos, ctx->item.count);
if (!layout_flex__resolve_line(ctx, line)) {
return false;
}
if (!layout_flex__place_line_items_main(ctx, line)) {
return false;
}
ctx->cross_size += line->cross_size;
if (ctx->main_size < line->main_size) {
ctx->main_size = line->main_size;
}
}
return true;
}
/**
* Align items on a line.
*
* \param[in] ctx Flex layout context
* \param[in] line Line to align items on
* \param[in] extra Extra line width in pixels
*/
static void layout_flex__place_line_items_cross(struct flex_ctx *ctx,
struct flex_line_data *line, int extra)
{
enum box_side cross_start = ctx->horizontal ? TOP : LEFT;
size_t item_count = line->first + line->count;
for (size_t i = line->first; i < item_count; i++) {
struct flex_item_data *item = &ctx->item.data[i];
struct box *b = item->box;
int cross_free_space;
int *box_size_cross;
int *box_pos_cross;
box_pos_cross = ctx->horizontal ? &b->y : &b->x;
box_size_cross = lh__box_size_cross_ptr(ctx->horizontal, b);
cross_free_space = line->cross_size + extra - *box_size_cross -
lh__delta_outer_cross(ctx->flex, b);
switch (lh__box_align_self(ctx->flex, b)) {
default:
/* Fall through. */
case CSS_ALIGN_SELF_STRETCH:
if (lh__box_size_cross_is_auto(ctx->horizontal, b)) {
*box_size_cross += cross_free_space;
/* Relayout children for stretch. */
if (!layout_flex_item(ctx, item, b->width)) {
return;
}
}
/* Fall through. */
case CSS_ALIGN_SELF_FLEX_START:
*box_pos_cross = ctx->flex->padding[cross_start] +
line->pos +
lh__non_auto_margin(b, cross_start) +
b->border[cross_start].width;
break;
case CSS_ALIGN_SELF_FLEX_END:
*box_pos_cross = ctx->flex->padding[cross_start] +
line->pos + cross_free_space +
lh__non_auto_margin(b, cross_start) +
b->border[cross_start].width;
break;
case CSS_ALIGN_SELF_BASELINE:
/* Fall through. */
case CSS_ALIGN_SELF_CENTER:
*box_pos_cross = ctx->flex->padding[cross_start] +
line->pos + cross_free_space / 2 +
lh__non_auto_margin(b, cross_start) +
b->border[cross_start].width;
break;
}
}
}
/**
* Place the lines and align the items on the line.
*
* \param[in] ctx Flex layout context
*/
static void layout_flex__place_lines(struct flex_ctx *ctx)
{
bool reversed = ctx->wrap == CSS_FLEX_WRAP_WRAP_REVERSE;
int line_pos = reversed ? ctx->cross_size : 0;
int post_multiplier = reversed ? 0 : 1;
int pre_multiplier = reversed ? -1 : 0;
int extra_remainder = 0;
int extra = 0;
if (ctx->available_cross != AUTO &&
ctx->available_cross > ctx->cross_size &&
ctx->line.count > 0) {
extra = ctx->available_cross - ctx->cross_size;
extra_remainder = extra % ctx->line.count;
extra /= ctx->line.count;
}
for (size_t i = 0; i < ctx->line.count; i++) {
struct flex_line_data *line = &ctx->line.data[i];
line_pos += pre_multiplier * line->cross_size;
line->pos = line_pos;
line_pos += post_multiplier * line->cross_size +
extra + extra_remainder;
layout_flex__place_line_items_cross(ctx, line,
extra + extra_remainder);
if (extra_remainder > 0) {
extra_remainder--;
}
}
}
/**
* Layout a flex container.
*
* \param[in] flex table to layout
* \param[in] available_width width of containing block
* \param[in] content memory pool for any new boxes
* \return true on success, false on memory exhaustion
*/
bool layout_flex(struct box *flex, int available_width,
html_content *content)
{
int max_height, min_height;
struct flex_ctx *ctx;
bool success = false;
ctx = layout_flex_ctx__create(content, flex);
if (ctx == NULL) {
return false;
}
NSLOG(flex, DEEPDEBUG, "box %p: %s, available_width %i, width: %i",
flex, ctx->horizontal ? "horizontal" : "vertical",
available_width, flex->width);
layout_find_dimensions(
ctx->unit_len_ctx, available_width, -1,
flex, flex->style, NULL, &flex->height,
NULL, NULL, &max_height, &min_height,
flex->margin, flex->padding, flex->border);
available_width = min(available_width, flex->width);
if (ctx->horizontal) {
ctx->available_main = available_width;
ctx->available_cross = ctx->flex->height;
} else {
ctx->available_main = ctx->flex->height;
ctx->available_cross = available_width;
}
NSLOG(flex, DEEPDEBUG, "box %p: available_main: %i",
flex, ctx->available_main);
NSLOG(flex, DEEPDEBUG, "box %p: available_cross: %i",
flex, ctx->available_cross);
layout_flex_ctx__populate_item_data(ctx, flex, available_width);
/* Place items onto lines. */
success = layout_flex__collect_items_into_lines(ctx);
if (!success) {
goto cleanup;
}
layout_flex__place_lines(ctx);
if (flex->height == AUTO) {
flex->height = ctx->horizontal ?
ctx->cross_size :
ctx->main_size;
}
if (flex->height != AUTO) {
if (max_height >= 0 && flex->height > max_height) {
flex->height = max_height;
}
if (min_height > 0 && flex->height < min_height) {
flex->height = min_height;
}
}
success = true;
cleanup:
layout_flex_ctx__destroy(ctx);
NSLOG(flex, DEEPDEBUG, "box %p: %s: w: %i, h: %i", flex,
success ? "success" : "failure",
flex->width, flex->height);
return success;
}