diff --git a/backend/app/views/spree/admin/promotions/rules/_minimum_quantity.html.erb b/backend/app/views/spree/admin/promotions/rules/_minimum_quantity.html.erb new file mode 100644 index 00000000000..e48fba7f1b7 --- /dev/null +++ b/backend/app/views/spree/admin/promotions/rules/_minimum_quantity.html.erb @@ -0,0 +1,5 @@ +
+ <% field_name = "#{param_prefix}[preferred_minimum_quantity]" %> + <%= label_tag field_name, promotion_rule.model_name.human %> + <%= number_field_tag field_name, promotion_rule.preferred_minimum_quantity, class: "fullwidth", min: 1 %> +
diff --git a/core/app/models/spree/promotion/rules/minimum_quantity.rb b/core/app/models/spree/promotion/rules/minimum_quantity.rb new file mode 100644 index 00000000000..409a4edabde --- /dev/null +++ b/core/app/models/spree/promotion/rules/minimum_quantity.rb @@ -0,0 +1,59 @@ +# frozen_string_literal: true + +module Spree + class Promotion + module Rules + # Promotion rule for ensuring an order contains a minimum quantity of + # applicable items. + # + # This promotion rule is only compatible with the "all" match policy. It + # doesn't make a lot of sense to use it without that policy as it reduces + # it to a simple quantity check across the entire order which would be + # better served by an item total rule. + class MinimumQuantity < PromotionRule + validates :preferred_minimum_quantity, numericality: { only_integer: true, greater_than: 0 } + + preference :minimum_quantity, :integer, default: 1 + + # What type of objects we should run our eligiblity checks against. In + # this case, our rule only applies to an entire order. + # + # @param promotable [Spree::Order,Spree::LineItem] + # @return [Boolean] true if promotable is a Spree::Order, false + # otherwise + def applicable?(promotable) + promotable.is_a?(Spree::Order) + end + + # Will look at all of the "applicable" line items in the order and + # determine if the sum of their quantity is greater than the minimum. + # + # "Applicable" items are ones where they pass the "actionable?" check of + # all rules on the promotion. (e.g.: Match product/taxon when one of + # those rules is present.) + # + # When false is returned, the reason will be included in the + # `eligibility_errors` object. + # + # @param order [Spree::Order] the order we want to check eligibility on + # @param _options [Hash] ignored + # @return [Boolean] true if promotion is eligible, false otherwise + def eligible?(order, _options = {}) + applicable_line_items = order.line_items.select do |line_item| + promotion.rules.all? { _1.actionable?(line_item) } + end + + if applicable_line_items.sum(&:quantity) < preferred_minimum_quantity + eligibility_errors.add( + :base, + eligibility_error_message(:quantity_less_than_minimum, count: preferred_minimum_quantity), + error_code: :quantity_less_than_minimum + ) + end + + eligibility_errors.empty? + end + end + end + end +end diff --git a/core/config/locales/en.yml b/core/config/locales/en.yml index 7237e213402..ee626bc9529 100644 --- a/core/config/locales/en.yml +++ b/core/config/locales/en.yml @@ -217,6 +217,8 @@ en: description: Order total meets these criteria spree/promotion/rules/landing_page: description: Customer must have visited the specified page + spree/promotion/rules/minimum_quantity: + description: Order contains minimum quantity of applicable items spree/promotion/rules/nth_order: description: Apply a promotion to every nth order a user has completed. form_text: 'Apply this promotion on the users Nth order: ' @@ -652,6 +654,7 @@ en: spree/promotion/rules/first_repeat_purchase_since: First Repeat Purchase Since spree/promotion/rules/item_total: Item Total spree/promotion/rules/landing_page: Landing Page + spree/promotion/rules/minimum_quantity: Minimum Quantity spree/promotion/rules/nth_order: Nth Order spree/promotion/rules/one_use_per_user: One Use Per User spree/promotion/rules/option_value: Option Value(s) @@ -1533,6 +1536,9 @@ en: no_user_or_email_specified: You need to login or provide your email before applying this coupon code. no_user_specified: You need to login before applying this coupon code. not_first_order: This coupon code can only be applied to your first order. + quantity_less_than_minimum: + one: "You need to add a least 1 applicable item to your order." + other: "You need to add a least %{count} applicable items to your order." email: Email empty: Empty empty_cart: Empty Cart diff --git a/core/lib/spree/app_configuration.rb b/core/lib/spree/app_configuration.rb index 49baffcc554..22988d6bf76 100644 --- a/core/lib/spree/app_configuration.rb +++ b/core/lib/spree/app_configuration.rb @@ -665,6 +665,7 @@ def environment Spree::Promotion::Rules::UserLoggedIn Spree::Promotion::Rules::OneUsePerUser Spree::Promotion::Rules::Taxon + Spree::Promotion::Rules::MinimumQuantity Spree::Promotion::Rules::NthOrder Spree::Promotion::Rules::OptionValue Spree::Promotion::Rules::FirstRepeatPurchaseSince diff --git a/core/spec/models/spree/promotion/rules/minimum_quantity_spec.rb b/core/spec/models/spree/promotion/rules/minimum_quantity_spec.rb new file mode 100644 index 00000000000..60c58df5467 --- /dev/null +++ b/core/spec/models/spree/promotion/rules/minimum_quantity_spec.rb @@ -0,0 +1,108 @@ +# frozen_string_literal: true + +RSpec.describe Spree::Promotion::Rules::MinimumQuantity do + subject(:quantity_rule) { described_class.new(preferred_minimum_quantity: 2) } + + describe "#valid?" do + let(:promotion) { build(:promotion) } + + before { promotion.rules << quantity_rule } + + it { is_expected.to be_valid } + + context "when minimum quantity is zero" do + subject(:quantity_rule) { described_class.new(preferred_minimum_quantity: 0) } + + it { is_expected.not_to be_valid } + end + end + + describe "#applicable?" do + subject { quantity_rule.applicable?(promotable) } + + context "when promotable is an order" do + let(:promotable) { Spree::Order.new } + + it { is_expected.to be true } + end + + context "when promotable is a line item" do + let(:promotable) { Spree::LineItem.new } + + it { is_expected.to be false } + end + end + + describe "#eligible?" do + subject { quantity_rule.eligible?(order) } + + let(:order) { + create( + :order_with_line_items, + line_items_count: line_items.length, + line_items_attributes: line_items + ) + } + let(:promotion) { build(:promotion) } + + before { promotion.rules << quantity_rule } + + context "when only the quantity rule is applied" do + context "when the quantity is less than the minimum" do + let(:line_items) { [{ quantity: 1 }] } + + it { is_expected.to be false } + end + + context "when the quantity is equal to the minimum" do + let(:line_items) { [{ quantity: 2 }] } + + it { is_expected.to be true } + end + + context "when the quantity is greater than the minimum" do + let(:line_items) { [{ quantity: 4 }] } + + it { is_expected.to be true } + end + end + + context "when another rule limits the applicable items" do + let(:carry_on) { create(:away_base_variant) } + let(:other_carry_on) { create(:away_base_variant) } + let(:everywhere_bag) { create(:product, :everywhere_bag).master } + + let(:product_rule) { + Spree::Promotion::Rules::Product.new( + products: [carry_on.product, other_carry_on.product], + preferred_match_policy: "any" + ) + } + + before { promotion.rules << product_rule } + + context "when the applicable quantity is less than the minimum" do + let(:line_items) do + [ + { variant: carry_on, quantity: 1 }, + { variant: everywhere_bag, quantity: 1 } + ] + end + + it { is_expected.to be false } + end + + context "when the applicable quantity is greater than the minimum" do + let(:line_items) do + [ + { variant: carry_on, quantity: 1 }, + { variant: other_carry_on, quantity: 1 }, + { variant: everywhere_bag, quantity: 1 } + ] + end + + it { is_expected.to be true } + end + end + end +end