/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /*************************************************************************** * layout.cc * * Sat Mar 21 15:12:36 CET 2015 * Copyright 2015 Bent Bisballe Nyeng * deva@aasimon.org ****************************************************************************/ /* * This file is part of DrumGizmo. * * DrumGizmo is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * DrumGizmo 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 Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with DrumGizmo; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. */ #include "layout.h" #include "widget.h" #include <algorithm> namespace GUI { LayoutItem::LayoutItem() : parent(nullptr) { } LayoutItem::~LayoutItem() { setLayoutParent(nullptr); // Will disconnect from layout if any. } void LayoutItem::setLayoutParent(Layout* p) { if(this->parent) { this->parent->removeItem(this); } this->parent = p; } Layout::Layout(LayoutItem* parent) : parent(parent) { auto widget = dynamic_cast<Widget*>(parent); if(widget) { CONNECT(widget, sizeChangeNotifier, this, &Layout::sizeChanged); } } void Layout::addItem(LayoutItem* item) { items.push_back(item); item->setLayoutParent(this); layout(); } void Layout::removeItem(LayoutItem* item) { auto new_end = std::remove(items.begin(), items.end(), item); items.erase(new_end, items.end()); layout(); } void Layout::sizeChanged(int width, int height) { layout(); } // // BoxLayout // BoxLayout::BoxLayout(LayoutItem* parent) : Layout(parent) { } void BoxLayout::setResizeChildren(bool resizeChildren) { this->resizeChildren = resizeChildren; layout(); } void BoxLayout::setSpacing(size_t spacing) { this->spacing = spacing; layout(); } // // VBoxLayout // VBoxLayout::VBoxLayout(LayoutItem* parent) : BoxLayout(parent) , align(HAlignment::center) { } void VBoxLayout::layout() { size_t y = 0; size_t w = parent->width(); // size_t h = parent->height() / items.size(); LayoutItemList::iterator i = items.begin(); while(i != items.end()) { LayoutItem* item = *i; if(resizeChildren) { auto num_items = items.size(); auto empty_space = (num_items - 1) * spacing; auto available_space = parent->height(); if(available_space >= empty_space) { auto item_height = (available_space - empty_space) / num_items; item->resize(w, item_height); } else { // TODO: Should this case be handled differently? item->resize(w, 0); } } size_t x = 0; switch(align) { case HAlignment::left: x = 0; break; case HAlignment::center: x = (w / 2) - (item->width() / 2); break; case HAlignment::right: x = w - item->width(); break; } item->move(x, y); y += item->height() + spacing; ++i; } } void VBoxLayout::setHAlignment(HAlignment alignment) { align = alignment; } // // HBoxLayout // HBoxLayout::HBoxLayout(LayoutItem* parent) : BoxLayout(parent), align(VAlignment::center) { } void HBoxLayout::layout() { if(items.empty()) { return; } // size_t w = parent->width() / items.size(); size_t h = parent->height(); size_t x = 0; LayoutItemList::iterator i = items.begin(); while(i != items.end()) { LayoutItem* item = *i; if(resizeChildren) { auto num_items = items.size(); auto empty_space = (num_items - 1) * spacing; auto available_space = parent->width(); if(available_space >= empty_space) { auto item_width = (available_space - empty_space) / num_items; item->resize(item_width, h); } else { // TODO: Should this case be handled differently? item->resize(0, h); } item->move(x, 0); } else { size_t y = 0; switch(align) { case VAlignment::top: y = 0; break; case VAlignment::center: y = (h / 2) - (item->height() / 2); break; case VAlignment::bottom: y = h - item->height(); break; } int diff = 0; // w - item->width(); item->move(x + diff / 2, y); } x += item->width() + spacing; ++i; } } void HBoxLayout::setVAlignment(VAlignment alignment) { align = alignment; } // // GridLayout // GridLayout::GridLayout(LayoutItem* parent, std::size_t number_of_columns, std::size_t number_of_rows) : BoxLayout(parent) , number_of_columns(number_of_columns) , number_of_rows(number_of_rows) { } void GridLayout::removeItem(LayoutItem* item) { // manually remove from grid_ranges as remove_if doesn't work on an // unordered_map. auto it = grid_ranges.begin(); while(it != grid_ranges.end()) { if(it->first == item) { it = grid_ranges.erase(it); } else { ++it; } } Layout::removeItem(item); } void GridLayout::layout() { if(grid_ranges.empty()) { return; } // Calculate cell sizes auto cell_size = calculateCellSize(); for(auto const& pair : grid_ranges) { auto& item = *pair.first; auto const& range = pair.second; moveAndResize(item, range, cell_size); } } void GridLayout::setPosition(LayoutItem* item, GridRange const& range) { grid_ranges[item] = range; } int GridLayout::lastUsedRow(int column) const { int last_row = -1; for (auto const& grid_range : grid_ranges) { auto const& range = grid_range.second; if (column >= range.column_begin && column < range.column_end) { last_row = std::max(last_row, range.row_end - 1); } } return last_row; } int GridLayout::lastUsedColumn(int row) const { int last_column = -1; for (auto const& grid_range : grid_ranges) { auto const& range = grid_range.second; if (row >= range.row_begin && row < range.row_end) { last_column = std::max(last_column, range.column_end - 1); } } return last_column; } auto GridLayout::calculateCellSize() const -> CellSize { auto empty_width = (number_of_columns - 1) * spacing; auto available_width = parent->width(); auto empty_height = (number_of_rows - 1) * spacing; auto available_height = parent->height(); CellSize cell_size; if(available_width > empty_width && available_height > empty_height) { cell_size.width = (available_width - empty_width) / number_of_columns; cell_size.height = (available_height - empty_height) / number_of_rows; } else { cell_size.width = 0; cell_size.height = 0; } return cell_size; } void GridLayout::moveAndResize( LayoutItem& item, GridRange const& range, CellSize cell_size) const { std::size_t x = range.column_begin * (cell_size.width + spacing); std::size_t y = range.row_begin * (cell_size.height + spacing); std::size_t column_count = (range.column_end - range.column_begin); std::size_t row_count = (range.row_end - range.row_begin); std::size_t width = column_count * (cell_size.width + spacing) - spacing; std::size_t height = row_count * (cell_size.height + spacing) - spacing; if(resizeChildren) { item.move(x, y); if(cell_size.width * cell_size.height != 0) { item.resize(width, height); } else { item.resize(0, 0); } } else { if(item.width() <= width && item.height() <= height) { item.move(x + (width - item.width()) / 2, y + (height - item.height()) / 2); } else { item.move(x, y); } } } } // GUI::