# -*- encoding: utf-8; frozen_string_literal: true -*- # #-- # This file is part of HexaPDF. # # HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby # Copyright (C) 2014-2024 Thomas Leitner # # HexaPDF is free software: you can redistribute it and/or modify it # under the terms of the GNU Affero General Public License version 3 as # published by the Free Software Foundation with the addition of the # following permission added to Section 15 as permitted in Section 7(a): # FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY # THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON # INFRINGEMENT OF THIRD PARTY RIGHTS. # # HexaPDF 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 Affero General Public # License for more details. # # You should have received a copy of the GNU Affero General Public License # along with HexaPDF. If not, see . # # The interactive user interfaces in modified source and object code # versions of HexaPDF must display Appropriate Legal Notices, as required # under Section 5 of the GNU Affero General Public License version 3. # # In accordance with Section 7(b) of the GNU Affero General Public # License, a covered work must retain the producer line in every PDF that # is created or manipulated using HexaPDF. # # If the GNU Affero General Public License doesn't fit your need, # commercial licenses are available at . #++ require 'hexapdf/layout/box' require 'hexapdf/layout/box_fitter' module HexaPDF module Layout # A ColumnBox arranges boxes in one or more columns. # # The number and width of the columns as well as the size of the gap between the columns can be # modified. Additionally, the contents can either fill the columns one after the other or the # columns can be made equally high. # # If the column box has padding and/or borders specified, they are handled like with any other # box. This means they are around all columns and their contents and are not used separately for # each column. # # The following style properties are used (additionally to those used by the parent class): # # Style#position:: # If this is set to :flow, the frames created for the columns will take the shape of the # frame into account. This also means that the +available_width+ and +available_height+ # arguments are ignored. class ColumnBox < Box # The child boxes of this ColumnBox. They need to be finalized before #fit is called. attr_reader :children # The columns definition. # # If the value is an array, it needs to contain the widths of the columns. The size of the # array determines the number of columns. Otherwise, if the value is an integer, the value # defines the number of equally sized columns, i.e. a value of +N+ is equal to [-1]*N. # # If a negative integer is used for the width, the column is auto-sized. Such columns split # the remaining width (after substracting the widths of the fixed columns) proportionally # among them. For example, if the definition is [-1, -2, -2], the first column is a fifth of # the width and the other columns are each two fifth of the width. # # Examples: # # #>pdf-composer # composer.box(:column, columns: 2, gaps: 10, # children: [composer.document.layout.lorem_ipsum_box]) # # --- # # #>pdf-composer # composer.box(:column, columns: [50, -2, -1], gaps: [10, 5], # children: [composer.document.layout.lorem_ipsum_box]) attr_reader :columns # The size of the gaps between the columns. # # This is an array containing the width of the gaps. If there are more gaps than numbers in # the array, the array is cycled. # # Examples: see #columns attr_reader :gaps # Determines whether the columns should all be equally high or not. # # Examples: # # #>pdf-composer # composer.box(:column, children: [composer.document.layout.lorem_ipsum_box]) # # --- # # #>pdf-composer # composer.box(:column, equal_height: false, # children: [composer.document.layout.lorem_ipsum_box]) attr_reader :equal_height # Creates a new ColumnBox object for the given child boxes in +children+. # # +columns+:: # # Can either simply integer specify the number of columns or be a full column definition # (see #columns for details). # # +gaps+:: # Can either be a simply integer specifying the width between two columns or a full gap # definition (see #gap for details). # # +equal_height+:: # If +true+, the #fit method tries to balance the columns in terms of their height. # Otherwise the columns are filled from the left. def initialize(children: [], columns: 2, gaps: 36, equal_height: true, **kwargs) super(**kwargs) @children = children @columns = (columns.kind_of?(Array) ? columns : [-1] * columns) @gaps = (gaps.kind_of?(Array) ? gaps : [gaps]) @equal_height = equal_height end # Returns +true+ as the 'position' style property value :flow is supported. def supports_position_flow? true end # Returns +true+ if no box was fitted into the columns. def empty? super && (!@box_fitter || @box_fitter.fit_results.empty?) end # Fits the column box into the current region of the frame. # # If the style property 'position' is set to :flow, the columns might not be rectangles but # arbitrary (sets of) polygons since the +frame+s shape is taken into account. def fit(available_width, available_height, frame) return false if @initial_height > available_height || @initial_width > available_width initial_fit_successful = (@equal_height && @columns.size > 1 ? nil : false) tries = 0 @width = if style.position == :flow (@initial_width > 0 ? @initial_width : frame.width) - reserved_width else (@initial_width > 0 ? @initial_width : available_width) - reserved_width end height = if style.position == :flow (@initial_height > 0 ? @initial_height : frame.height) - reserved_height else (@initial_height > 0 ? @initial_height : available_height) - reserved_height end columns = calculate_columns(@width) return false if columns.empty? left = (style.position == :flow ? frame.left : frame.x) + reserved_width_left top = (style.position == :flow ? frame.bottom + frame.height : frame.y) - reserved_height_top successful_height = height unsuccessful_height = 0 while true @box_fitter = BoxFitter.new columns.each do |col_x, column_width| column_left = left + col_x column_bottom = top - height if style.position == :flow rect = Geom2D::Polygon([column_left, column_bottom], [column_left + column_width, column_bottom], [column_left + column_width, column_bottom + height], [column_left, column_bottom + height]) shape = Geom2D::Algorithms::PolygonOperation.run(frame.shape, rect, :intersection) end column_frame = frame.child_frame(column_left, column_bottom, column_width, height, shape: shape, box: self) @box_fitter << column_frame end children.each {|box| @box_fitter.fit(box) } fit_successful = @box_fitter.fit_successful? initial_fit_successful = fit_successful if initial_fit_successful.nil? if fit_successful successful_height = height if successful_height > height elsif unsuccessful_height < height unsuccessful_height = height end break if !initial_fit_successful || tries > 40 || (fit_successful && successful_height - unsuccessful_height < 10) height = if successful_height - unsuccessful_height <= 5 successful_height else (successful_height + unsuccessful_height) / 2.0 end tries += 1 end @width = columns[-1].sum + reserved_width @height = (@initial_height > 0 ? @initial_height : @box_fitter.content_heights.max + reserved_height) @draw_pos_x = frame.x + reserved_width_left @draw_pos_y = frame.y - @height + reserved_height_bottom @box_fitter.fit_successful? end private # Calculates the x-coordinates and widths of all columns based on the given total available # width. # # If it is not possible to fit all columns into the given +width+, an empty array is returned. def calculate_columns(width) number_of_columns = @columns.size gaps = @gaps.cycle.take(number_of_columns - 1) fixed_width, variable_width = @columns.partition(&:positive?).map {|c| c.sum(&:abs) } rest_width = width - fixed_width - gaps.sum return [] if rest_width <= 0 variable_width_unit = rest_width / variable_width.to_f position = 0 @columns.map.with_index do |column, index| result = if column > 0 [position, column] else [position, column.abs * variable_width_unit] end position += result[1] + (gaps[index] || 0) result end end # Splits the content of the column box. This method is called from Box#split. def split_content(_available_width, _available_height, _frame) box = create_split_box box.instance_variable_set(:@children, @box_fitter.remaining_boxes) [self, box] end # Draws the child boxes onto the canvas at position [x, y]. def draw_content(canvas, x, y) if style.position != :flow && (x != @draw_pos_x || y != @draw_pos_y) canvas.translate(x - @draw_pos_x, y - @draw_pos_y) do @box_fitter.fit_results.each {|result| result.draw(canvas) } end else @box_fitter.fit_results.each {|result| result.draw(canvas) } end end end end end