#if defined(Hiro_HorizontalLayout) auto mHorizontalLayout::alignment() const -> maybe { return state.alignment; } auto mHorizontalLayout::append(sSizable sizable, Size size, float spacing) -> type& { for(auto& cell : state.cells) { if(cell->state.sizable == sizable) return *this; } HorizontalLayoutCell cell; cell->setSizable(sizable); cell->setSize(size); cell->setSpacing(spacing); cell->setParent(this, cellCount()); state.cells.append(cell); return synchronize(); } auto mHorizontalLayout::cell(uint position) const -> HorizontalLayoutCell { return state.cells(position, {}); } auto mHorizontalLayout::cell(sSizable sizable) const -> HorizontalLayoutCell { for(auto& cell : state.cells) { if(cell->state.sizable == sizable) return cell; } return {}; } auto mHorizontalLayout::cellCount() const -> uint { return state.cells.size(); } auto mHorizontalLayout::destruct() -> void { for(auto& cell : state.cells) cell->destruct(); mSizable::destruct(); } auto mHorizontalLayout::minimumSize() const -> Size { float width = 0; for(auto index : range(cellCount())) { auto cell = this->cell(index); if(cell.size().width() == Size::Minimum || cell.size().width() == Size::Maximum) { width += cell.sizable().minimumSize().width(); } else { width += cell.size().width(); } if(index != cellCount() - 1) width += cell.spacing(); } float height = 0; for(auto index : range(cellCount())) { auto cell = this->cell(index); if(cell.size().height() == Size::Minimum || cell.size().height() == Size::Maximum) { height = max(height, cell.sizable().minimumSize().height()); continue; } height = max(height, cell.size().height()); } return { padding().x() + width + padding().width(), padding().y() + height + padding().height() }; } auto mHorizontalLayout::padding() const -> Geometry { return state.padding; } auto mHorizontalLayout::remove(sSizable sizable) -> type& { for(auto& cell : state.cells) { if(cell->state.sizable == sizable) return remove(cell); } return *this; } auto mHorizontalLayout::remove(sHorizontalLayoutCell cell) -> type& { if(cell->parent() != this) return *this; auto offset = cell->offset(); cell->setParent(); state.cells.remove(offset); for(uint n : range(offset, cellCount())) state.cells[n]->adjustOffset(-1); return synchronize(); } auto mHorizontalLayout::reset() -> type& { while(state.cells) remove(state.cells.right()); return synchronize(); } auto mHorizontalLayout::setAlignment(maybe alignment) -> type& { state.alignment = alignment; return synchronize(); } auto mHorizontalLayout::setEnabled(bool enabled) -> type& { mSizable::setEnabled(enabled); for(auto& cell : state.cells) cell.sizable().setEnabled(cell.sizable().enabled()); return *this; } auto mHorizontalLayout::setFont(const Font& font) -> type& { mSizable::setFont(font); for(auto& cell : state.cells) cell.sizable().setFont(cell.sizable().font()); return *this; } auto mHorizontalLayout::setGeometry(Geometry geometry) -> type& { mSizable::setGeometry(geometry); if(!visible(true)) return *this; geometry.setX(geometry.x() + padding().x()); geometry.setY(geometry.y() + padding().y()); geometry.setWidth (geometry.width() - padding().x() - padding().width()); geometry.setHeight(geometry.height() - padding().y() - padding().height()); vector widths; widths.resize(cellCount()); uint maximumWidths = 0; for(auto index : range(cellCount())) { auto cell = this->cell(index); float width = 0; if(cell.size().width() == Size::Maximum) { width = Size::Maximum; maximumWidths++; } else if(cell.size().width() == Size::Minimum) { width = cell.sizable().minimumSize().width(); } else { width = cell.size().width(); } widths[index] = width; } float fixedWidth = 0; for(uint index : range(state.cells.size())) { if(widths[index] != Size::Maximum) fixedWidth += widths[index]; if(index != cellCount() - 1) fixedWidth += cell(index).spacing(); } float maximumWidth = (geometry.width() - fixedWidth) / maximumWidths; for(auto& width : widths) { if(width == Size::Maximum) width = maximumWidth; } float height = 0; for(auto index : range(cellCount())) { auto cell = this->cell(index); if(cell.size().height() == Size::Maximum) { height = geometry.height(); break; } else if(cell.size().height() == Size::Minimum) { height = max(height, cell.sizable().minimumSize().height()); } else { height = max(height, cell.size().height()); } } float geometryX = geometry.x(); float geometryY = geometry.y(); for(auto index : range(cellCount())) { float geometryWidth = widths[index]; float geometryHeight = height; auto cell = this->cell(index); auto alignment = cell.alignment(); if(!alignment) alignment = this->alignment(); if(!alignment) alignment = 0.5; float cellWidth = geometryWidth; float cellHeight = cell.size().height(); if(cellHeight == Size::Minimum) cellHeight = cell.sizable()->minimumSize().height(); if(cellHeight == Size::Maximum) cellHeight = geometryHeight; float cellX = geometryX; float cellY = geometryY + alignment() * (geometryHeight - cellHeight); cell.sizable().setGeometry({cellX, cellY, cellWidth, cellHeight}); geometryX += geometryWidth + cell.spacing(); } return *this; } auto mHorizontalLayout::setPadding(Geometry padding) -> type& { state.padding = padding; return synchronize(); } auto mHorizontalLayout::setParent(mObject* parent, int offset) -> type& { for(auto& cell : reverse(state.cells)) cell->destruct(); mSizable::setParent(parent, offset); for(auto& cell : state.cells) cell->setParent(this, cell->offset()); return *this; } auto mHorizontalLayout::setSpacing(float spacing) -> type& { state.spacing = spacing; return synchronize(); } auto mHorizontalLayout::setVisible(bool visible) -> type& { mSizable::setVisible(visible); for(auto& cell : state.cells) cell.sizable().setVisible(cell.sizable().visible()); return synchronize(); } auto mHorizontalLayout::spacing() const -> float { return state.spacing; } auto mHorizontalLayout::synchronize() -> type& { setGeometry(geometry()); return *this; } // auto mHorizontalLayoutCell::alignment() const -> maybe { return state.alignment; } auto mHorizontalLayoutCell::destruct() -> void { if(auto& sizable = state.sizable) sizable->destruct(); mObject::destruct(); } auto mHorizontalLayoutCell::setAlignment(maybe alignment) -> type& { state.alignment = alignment; return synchronize(); } auto mHorizontalLayoutCell::setEnabled(bool enabled) -> type& { mObject::setEnabled(enabled); state.sizable->setEnabled(state.sizable->enabled()); return *this; } auto mHorizontalLayoutCell::setFont(const Font& font) -> type& { mObject::setFont(font); state.sizable->setFont(state.sizable->font()); return *this; } auto mHorizontalLayoutCell::setParent(mObject* parent, int offset) -> type& { state.sizable->destruct(); mObject::setParent(parent, offset); state.sizable->setParent(this, 0); return *this; } auto mHorizontalLayoutCell::setSizable(sSizable sizable) -> type& { state.sizable = sizable; return synchronize(); } auto mHorizontalLayoutCell::setSize(Size size) -> type& { state.size = size; return synchronize(); } auto mHorizontalLayoutCell::setSpacing(float spacing) -> type& { state.spacing = spacing; return synchronize(); } auto mHorizontalLayoutCell::setVisible(bool visible) -> type& { mObject::setVisible(visible); state.sizable->setVisible(state.sizable->visible()); return *this; } auto mHorizontalLayoutCell::sizable() const -> Sizable { return state.sizable ? state.sizable : Sizable(); } auto mHorizontalLayoutCell::size() const -> Size { return state.size; } auto mHorizontalLayoutCell::spacing() const -> float { return state.spacing; } auto mHorizontalLayoutCell::synchronize() -> type& { if(auto parent = this->parent()) { if(auto horizontalLayout = dynamic_cast(parent)) { horizontalLayout->synchronize(); } } return *this; } #endif