UNPKG

@wordpress/block-library

Version:
80 lines (68 loc) 2.43 kB
/** * External dependencies */ import { View, StyleSheet } from 'react-native'; /** * WordPress dependencies */ import { Children } from '@wordpress/element'; /** * Internal dependencies */ import styles from './tiles-styles.scss'; function Tiles( props ) { const { columns, children, spacing = 10, style } = props; const { compose } = StyleSheet; const tileCount = Children.count( children ); const lastTile = tileCount - 1; const lastRow = Math.floor( lastTile / columns ); const wrappedChildren = Children.map( children, ( child, index ) => { /** * Since we don't have `calc()`, we must calculate our spacings here in * order to preserve even spacing between tiles and equal width for tiles * in a given row. * * In order to ensure equal sizing of tile contents, we distribute the * spacing such that each tile has an equal "share" of the fixed spacing. To * keep the tiles properly aligned within their rows, we calculate the left * and right paddings based on the tile's relative position within the row. * * Note: we use padding instead of margins so that the fixed spacing is * included within the relative spacing (i.e. width percentage), and * wrapping behavior is preserved. * * - The left most tile in a row must have left padding of zero. * - The right most tile in a row must have a right padding of zero. * * The values of these left and right paddings are interpolated for tiles in * between. The right padding is complementary with the left padding of the * next tile (i.e. the right padding of [tile n] + the left padding of * [tile n + 1] will be equal for all tiles except the last one in a given * row). */ const row = Math.floor( index / columns ); const rowLength = row === lastRow ? ( lastTile % columns ) + 1 : columns; const indexInRow = index % columns; return ( <View style={ [ styles.tileStyle, { width: `${ 100 / rowLength }%`, paddingLeft: spacing * ( indexInRow / rowLength ), paddingRight: spacing * ( 1 - ( indexInRow + 1 ) / rowLength ), paddingTop: row === 0 ? 0 : spacing / 2, paddingBottom: row === lastRow ? 0 : spacing / 2, }, ] } > { child } </View> ); } ); const containerStyle = compose( styles.containerStyle, style ); return <View style={ containerStyle }>{ wrappedChildren }</View>; } export default Tiles;