# frozen_string_literal: true module Physical class Package extend Forwardable attr_reader :container, :items, :void_fill_density, :id def initialize(id: nil, container: nil, items: [], void_fill_density: Measured::Density(0, :g_ml), dimensions: nil, weight: nil, properties: {}) @id = id || SecureRandom.uuid @void_fill_density = Types::Density[void_fill_density] @container = container || Physical::Box.new(dimensions: dimensions || [], weight: weight || Measured::Weight(0, :g), properties: properties) @items = Set[*items] end delegate [:dimensions, :width, :length, :height, :properties, :volume] => :container def <<(other) @items.add(other) end alias_method :add, :<< def >>(other) @items.delete(other) end alias_method :delete, :>> def weight container.weight + items_weight + void_fill_weight end # Cost is optional. We will only return an aggregate if all items # have cost defined. Otherwise we will retun nil. # @return Money def items_value items_cost = items.map(&:cost) items_cost.reduce(&:+) if items_cost.compact.size == items_cost.size end # @return [Measured::Weight] def items_weight items.map(&:weight).reduce(Measured::Weight(0, :g), &:+) end def void_fill_weight return Measured::Weight(0, :g) if container.volume.value.infinite? Measured::Weight(void_fill_density.convert_to(:g_ml).value * remaining_volume.convert_to(:ml).value, :g) end # @return [Measured::Volume] def used_volume items.map(&:volume).reduce(Measured::Volume(0, :ml), &:+) end def remaining_volume container.inner_volume - used_volume end def density return Measured::Density(Float::INFINITY, :g_ml) if container.volume.value.zero? return Measured::Density(0.0, :g_ml) if container.volume.value.infinite? Measured::Density(weight.convert_to(:g).value / container.volume.convert_to(:ml).value, :g_ml) end end end