2009-01-08 16:33:27 +01:00
|
|
|
# An OrderArticle represents a single Article that is part of an Order.
|
2009-01-06 11:49:19 +01:00
|
|
|
class OrderArticle < ActiveRecord::Base
|
2009-01-06 15:45:19 +01:00
|
|
|
|
2009-01-06 11:49:19 +01:00
|
|
|
belongs_to :order
|
|
|
|
belongs_to :article
|
2009-01-29 01:57:51 +01:00
|
|
|
belongs_to :article_price
|
2009-01-06 11:49:19 +01:00
|
|
|
has_many :group_order_articles, :dependent => :destroy
|
|
|
|
|
2009-02-09 20:12:56 +01:00
|
|
|
validates_presence_of :order_id, :article_id
|
2009-01-29 21:28:22 +01:00
|
|
|
validate :article_and_price_exist
|
2012-06-21 17:19:00 +02:00
|
|
|
validates_uniqueness_of :article_id, scope: :order_id
|
2009-01-06 11:49:19 +01:00
|
|
|
|
2011-05-11 12:21:21 +02:00
|
|
|
scope :ordered, :conditions => "units_to_order >= 1"
|
2009-01-29 01:57:51 +01:00
|
|
|
|
2012-06-21 17:19:00 +02:00
|
|
|
before_create :init_from_balancing
|
|
|
|
after_destroy :update_ordergroup_prices
|
2009-01-29 21:28:22 +01:00
|
|
|
|
2009-02-03 21:14:48 +01:00
|
|
|
# This method returns either the ArticlePrice or the Article
|
|
|
|
# The first will be set, when the the order is finished
|
2009-01-29 01:57:51 +01:00
|
|
|
def price
|
|
|
|
article_price || article
|
|
|
|
end
|
|
|
|
|
2009-01-29 21:28:22 +01:00
|
|
|
# Count quantities of belonging group_orders.
|
|
|
|
# In balancing this can differ from ordered (by supplier) quantity for this article.
|
|
|
|
def group_orders_sum
|
2009-02-04 16:41:01 +01:00
|
|
|
quantity = group_order_articles.collect(&:result).sum
|
2009-01-29 21:28:22 +01:00
|
|
|
{:quantity => quantity, :price => quantity * price.fc_price}
|
|
|
|
end
|
|
|
|
|
2009-02-03 21:14:48 +01:00
|
|
|
# Update quantity/tolerance/units_to_order from group_order_articles
|
|
|
|
def update_results!
|
2009-02-06 16:26:35 +01:00
|
|
|
if order.open?
|
|
|
|
quantity = group_order_articles.collect(&:quantity).sum
|
|
|
|
tolerance = group_order_articles.collect(&:tolerance).sum
|
|
|
|
update_attributes(:quantity => quantity, :tolerance => tolerance,
|
|
|
|
:units_to_order => calculate_units_to_order(quantity, tolerance))
|
|
|
|
elsif order.finished?
|
|
|
|
update_attribute(:units_to_order, group_order_articles.collect(&:result).sum)
|
|
|
|
end
|
2009-02-03 21:14:48 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
# Returns how many units of the belonging article need to be ordered given the specified order quantity and tolerance.
|
|
|
|
# This is determined by calculating how many units can be ordered from the given order quantity, using
|
|
|
|
# the tolerance to order an additional unit if the order quantity is not quiet sufficient.
|
|
|
|
# There must always be at least one item in a unit that is an ordered quantity (no units are ever entirely
|
|
|
|
# filled by tolerance items only).
|
|
|
|
#
|
|
|
|
# Example:
|
|
|
|
#
|
|
|
|
# unit_quantity | quantity | tolerance | calculate_units_to_order
|
|
|
|
# --------------+----------+-----------+-----------------------
|
|
|
|
# 4 | 0 | 2 | 0
|
|
|
|
# 4 | 0 | 5 | 0
|
|
|
|
# 4 | 2 | 2 | 1
|
|
|
|
# 4 | 4 | 2 | 1
|
|
|
|
# 4 | 4 | 4 | 1
|
|
|
|
# 4 | 5 | 3 | 2
|
|
|
|
# 4 | 5 | 4 | 2
|
|
|
|
#
|
|
|
|
def calculate_units_to_order(quantity, tolerance = 0)
|
|
|
|
unit_size = price.unit_quantity
|
|
|
|
units = quantity / unit_size
|
|
|
|
remainder = quantity % unit_size
|
|
|
|
units += ((remainder > 0) && (remainder + tolerance >= unit_size) ? 1 : 0)
|
|
|
|
end
|
|
|
|
|
2009-03-22 11:58:01 +01:00
|
|
|
# Calculate price for ordered quantity.
|
|
|
|
def total_price
|
2009-04-18 18:08:33 +02:00
|
|
|
units_to_order * price.unit_quantity * price.price
|
2009-03-22 11:58:01 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
# Calculate gross price for ordered qunatity.
|
|
|
|
def total_gross_price
|
2009-04-18 18:08:33 +02:00
|
|
|
units_to_order * price.unit_quantity * price.gross_price
|
2009-03-22 11:58:01 +01:00
|
|
|
end
|
|
|
|
|
2009-02-04 16:41:01 +01:00
|
|
|
def ordered_quantities_equal_to_group_orders?
|
|
|
|
(units_to_order * price.unit_quantity) == group_orders_sum[:quantity]
|
|
|
|
end
|
|
|
|
|
2009-02-09 20:12:56 +01:00
|
|
|
# Updates order_article and belongings during balancing process
|
|
|
|
def update_article_and_price!(article_attributes, price_attributes, order_article_attributes)
|
|
|
|
OrderArticle.transaction do
|
|
|
|
# Updates article
|
|
|
|
article.update_attributes!(article_attributes)
|
|
|
|
|
|
|
|
article_price.attributes = price_attributes
|
|
|
|
if article_price.changed?
|
|
|
|
# Creates a new article_price if neccessary
|
|
|
|
price = build_article_price(price_attributes)
|
|
|
|
price.created_at = order.ends
|
|
|
|
price.save!
|
|
|
|
|
|
|
|
# Updates ordergroup values
|
|
|
|
group_order_articles.each { |goa| goa.group_order.update_price! }
|
|
|
|
end
|
|
|
|
|
|
|
|
# Updates units_to_order
|
|
|
|
self.attributes = order_article_attributes
|
|
|
|
self.save!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2011-06-19 19:56:04 +02:00
|
|
|
# Units missing for the next full unit_quantity of the article
|
|
|
|
def missing_units
|
|
|
|
units = article.unit_quantity - ((quantity % article.unit_quantity) + tolerance)
|
|
|
|
units = 0 if units < 0
|
|
|
|
units
|
|
|
|
end
|
|
|
|
|
2009-01-06 11:49:19 +01:00
|
|
|
private
|
|
|
|
|
2009-01-29 21:28:22 +01:00
|
|
|
def article_and_price_exist
|
|
|
|
errors.add(:article, "muss angegeben sein und einen aktuellen Preis haben") if !(article = Article.find(article_id)) || article.fc_price.nil?
|
|
|
|
end
|
|
|
|
|
2012-06-21 17:19:00 +02:00
|
|
|
# Associate with current article price if created in a finished order
|
|
|
|
def init_from_balancing
|
|
|
|
if order.present? and order.finished?
|
|
|
|
self.article_price = article.article_prices.first
|
|
|
|
self.units_to_order = 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
#TODO: Delayed job maybe??
|
|
|
|
def update_ordergroup_prices
|
|
|
|
group_order_articles.each { |goa| goa.group_order.update_price! }
|
|
|
|
end
|
|
|
|
|
2009-01-06 11:49:19 +01:00
|
|
|
end
|
2011-05-07 21:55:24 +02:00
|
|
|
|
|
|
|
# == Schema Information
|
|
|
|
#
|
|
|
|
# Table name: order_articles
|
|
|
|
#
|
|
|
|
# id :integer(4) not null, primary key
|
|
|
|
# order_id :integer(4) default(0), not null
|
|
|
|
# article_id :integer(4) default(0), not null
|
|
|
|
# quantity :integer(4) default(0), not null
|
|
|
|
# tolerance :integer(4) default(0), not null
|
|
|
|
# units_to_order :integer(4) default(0), not null
|
|
|
|
# lock_version :integer(4) default(0), not null
|
|
|
|
# article_price_id :integer(4)
|
|
|
|
#
|
|
|
|
|