spree_core 3.0.10 → 3.1.0.rc1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/app/assets/javascripts/spree.js.coffee.erb +12 -3
- data/app/helpers/spree/base_helper.rb +4 -1
- data/app/helpers/spree/products_helper.rb +37 -6
- data/app/mailers/spree/base_mailer.rb +11 -2
- data/app/models/concerns/spree/adjustment_source.rb +3 -10
- data/app/models/concerns/spree/default_price.rb +7 -1
- data/app/models/concerns/spree/named_type.rb +1 -1
- data/app/models/concerns/spree/user_api_authentication.rb +7 -1
- data/app/models/concerns/spree/user_methods.rb +47 -0
- data/app/models/concerns/spree/user_reporting.rb +2 -2
- data/app/models/concerns/spree/vat_price_calculation.rb +47 -0
- data/app/models/spree/address.rb +8 -7
- data/app/models/spree/adjustable/adjuster/base.rb +25 -0
- data/app/models/spree/adjustable/adjuster/promotion.rb +41 -0
- data/app/models/spree/adjustable/adjuster/tax.rb +26 -0
- data/app/models/spree/adjustable/adjustments_updater.rb +22 -45
- data/app/models/spree/adjustment.rb +8 -10
- data/app/models/spree/app_configuration.rb +5 -2
- data/app/models/spree/base.rb +4 -0
- data/app/models/spree/calculator.rb +0 -5
- data/app/models/spree/calculator/default_tax.rb +2 -10
- data/app/models/spree/classification.rb +7 -3
- data/app/models/spree/country.rb +14 -3
- data/app/models/spree/credit_card.rb +21 -31
- data/app/models/spree/customer_return.rb +7 -15
- data/app/models/spree/gateway.rb +7 -6
- data/app/models/spree/image.rb +1 -1
- data/app/models/spree/inventory_unit.rb +9 -6
- data/app/models/spree/legacy_user.rb +1 -6
- data/app/models/spree/line_item.rb +69 -68
- data/app/models/spree/log_entry.rb +1 -4
- data/app/models/spree/option_type.rb +15 -6
- data/app/models/spree/option_type_prototype.rb +9 -0
- data/app/models/spree/option_value.rb +11 -3
- data/app/models/spree/option_value_variant.rb +6 -0
- data/app/models/spree/order.rb +113 -64
- data/app/models/spree/order/checkout.rb +8 -11
- data/app/models/spree/order/currency_updater.rb +1 -1
- data/app/models/spree/order/store_credit.rb +96 -0
- data/app/models/spree/order_contents.rb +6 -1
- data/app/models/spree/order_inventory.rb +4 -8
- data/app/models/spree/order_promotion.rb +6 -0
- data/app/models/spree/order_updater.rb +2 -7
- data/app/models/spree/payment.rb +46 -19
- data/app/models/spree/payment/gateway_options.rb +8 -4
- data/app/models/spree/payment_method.rb +12 -13
- data/app/models/spree/payment_method/store_credit.rb +130 -0
- data/app/models/spree/preference.rb +1 -1
- data/app/models/spree/price.rb +16 -6
- data/app/models/spree/product.rb +52 -49
- data/app/models/spree/product/scopes.rb +7 -2
- data/app/models/spree/product_option_type.rb +7 -2
- data/app/models/spree/product_promotion_rule.rb +9 -0
- data/app/models/spree/product_property.rb +8 -10
- data/app/models/spree/promotion.rb +19 -19
- data/app/models/spree/promotion/rules/product.rb +3 -1
- data/app/models/spree/promotion/rules/taxon.rb +2 -1
- data/app/models/spree/promotion/rules/user.rb +4 -4
- data/app/models/spree/promotion_action.rb +3 -3
- data/app/models/spree/promotion_category.rb +1 -1
- data/app/models/spree/promotion_rule_taxon.rb +9 -0
- data/app/models/spree/promotion_rule_user.rb +9 -0
- data/app/models/spree/property.rb +2 -1
- data/app/models/spree/property_prototype.rb +9 -0
- data/app/models/spree/prototype.rb +8 -3
- data/app/models/spree/prototype_taxon.rb +9 -0
- data/app/models/spree/refund.rb +10 -7
- data/app/models/spree/refund_reason.rb +1 -1
- data/app/models/spree/reimbursement.rb +12 -16
- data/app/models/spree/reimbursement_type/reimbursement_helpers.rb +23 -6
- data/app/models/spree/reimbursement_type/store_credit.rb +28 -0
- data/app/models/spree/return_authorization.rb +8 -13
- data/app/models/spree/return_authorization_reason.rb +1 -1
- data/app/models/spree/return_item.rb +13 -12
- data/app/models/spree/return_item/eligibility_validator/time_since_purchase.rb +1 -1
- data/app/models/spree/role.rb +3 -2
- data/app/models/spree/role_user.rb +6 -0
- data/app/models/spree/shipment.rb +18 -23
- data/app/models/spree/shipment_handler.rb +2 -2
- data/app/models/spree/shipping_category.rb +6 -3
- data/app/models/spree/shipping_method.rb +11 -10
- data/app/models/spree/shipping_method_zone.rb +6 -0
- data/app/models/spree/shipping_rate.rb +16 -29
- data/app/models/spree/state.rb +3 -2
- data/app/models/spree/state_change.rb +1 -1
- data/app/models/spree/stock/content_item.rb +10 -12
- data/app/models/spree/stock/coordinator.rb +13 -14
- data/app/models/spree/stock/estimator.rb +28 -30
- data/app/models/spree/stock/inventory_unit_builder.rb +1 -9
- data/app/models/spree/stock/packer.rb +1 -1
- data/app/models/spree/stock/quantifier.rb +5 -5
- data/app/models/spree/stock/splitter/backordered.rb +2 -2
- data/app/models/spree/stock_item.rb +12 -18
- data/app/models/spree/stock_location.rb +4 -7
- data/app/models/spree/stock_movement.rb +11 -9
- data/app/models/spree/stock_transfer.rb +11 -12
- data/app/models/spree/store.rb +14 -6
- data/app/models/spree/store_credit.rb +252 -0
- data/app/models/spree/store_credit_category.rb +22 -0
- data/app/models/spree/store_credit_event.rb +38 -0
- data/app/models/spree/store_credit_type.rb +6 -0
- data/app/models/spree/tax_category.rb +3 -8
- data/app/models/spree/tax_rate.rb +56 -122
- data/app/models/spree/taxon.rb +11 -5
- data/app/models/spree/tracker.rb +12 -1
- data/app/models/spree/validations/db_maximum_length_validator.rb +2 -1
- data/app/models/spree/variant.rb +82 -50
- data/app/models/spree/zone.rb +21 -17
- data/app/models/spree/zone_member.rb +6 -0
- data/app/validators/db_maximum_length_validator.rb +11 -0
- data/app/views/layouts/spree/base_mailer.html.erb +38 -781
- data/app/views/spree/order_mailer/_adjustment.html.erb +8 -0
- data/app/views/spree/order_mailer/_subtotal.html.erb +8 -0
- data/app/views/spree/order_mailer/_total.html.erb +8 -0
- data/app/views/spree/order_mailer/cancel_email.html.erb +13 -28
- data/app/views/spree/order_mailer/cancel_email.text.erb +1 -1
- data/app/views/spree/order_mailer/confirm_email.html.erb +49 -63
- data/app/views/spree/order_mailer/confirm_email.text.erb +1 -1
- data/app/views/spree/shared/_base_mailer_header.html.erb +5 -7
- data/app/views/spree/shared/_base_mailer_stylesheets.html.erb +777 -0
- data/app/views/spree/shared/_error_messages.html.erb +1 -1
- data/app/views/spree/shared/_mailer_line_item.html.erb +12 -0
- data/app/views/spree/shipment_mailer/shipped_email.html.erb +21 -14
- data/app/views/spree/shipment_mailer/shipped_email.text.erb +3 -3
- data/config/initializers/user_class_extensions.rb +7 -38
- data/config/locales/en.yml +113 -13
- data/config/routes.rb +7 -0
- data/db/default/spree/default_reimbursement_type.rb +1 -1
- data/db/default/spree/zones.rb +4 -5
- data/db/migrate/20150118210639_create_spree_store_credits.rb +24 -0
- data/db/migrate/20150118211500_create_spree_store_credit_categories.rb +8 -0
- data/db/migrate/20150118212051_create_spree_store_credit_events.rb +17 -0
- data/db/migrate/20150118212101_create_spree_store_credit_types.rb +10 -0
- data/db/migrate/20150314013438_add_missing_indexes.rb +25 -0
- data/db/migrate/20150317174308_remove_duplicated_indexes_from_multi_columns.rb +18 -0
- data/db/migrate/20150324104002_remove_user_index_from_spree_state_changes.rb +14 -0
- data/db/migrate/20150522071831_add_position_to_spree_payment_methods.rb +5 -0
- data/db/migrate/20150626181949_add_taxable_adjustment_total_to_line_item.rb +19 -0
- data/db/migrate/20150627090949_migrate_payment_methods_display.rb +12 -0
- data/db/migrate/20150714154102_spree_payment_method_store_credits.rb +12 -0
- data/db/migrate/20150726141425_rename_has_and_belongs_to_associations_to_model_names.rb +18 -0
- data/db/migrate/20150727191614_spree_store_credit_types.rb +11 -0
- data/db/migrate/20150819154308_add_discontinued_to_products_and_variants.rb +68 -0
- data/db/migrate/20151220072838_remove_shipping_method_id_from_spree_orders.rb +13 -0
- data/db/migrate/20160207191757_add_id_column_to_earlier_habtm_tables.rb +16 -0
- data/db/migrate/20160219165458_add_indexes.rb +14 -0
- data/lib/generators/spree/dummy/templates/rails/database.yml +31 -24
- data/lib/generators/spree/dummy/templates/rails/test.rb +2 -1
- data/lib/spree/core.rb +16 -0
- data/lib/spree/core/controller_helpers/auth.rb +1 -1
- data/lib/spree/core/controller_helpers/common.rb +3 -3
- data/lib/spree/core/controller_helpers/order.rb +6 -5
- data/lib/spree/core/controller_helpers/search.rb +1 -1
- data/lib/spree/core/controller_helpers/store.rb +29 -0
- data/lib/spree/core/delegate_belongs_to.rb +2 -2
- data/lib/spree/core/engine.rb +30 -25
- data/lib/spree/core/environment.rb +1 -1
- data/lib/spree/core/importer/order.rb +37 -40
- data/lib/spree/core/number_generator.rb +52 -0
- data/lib/spree/core/product_filters.rb +1 -1
- data/lib/spree/core/search/base.rb +4 -3
- data/lib/spree/core/version.rb +1 -1
- data/lib/spree/localized_number.rb +3 -1
- data/lib/spree/permitted_attributes.rb +5 -2
- data/lib/spree/testing_support/common_rake.rb +3 -3
- data/lib/spree/testing_support/factories.rb +3 -3
- data/lib/spree/testing_support/factories/address_factory.rb +1 -1
- data/lib/spree/testing_support/factories/country_factory.rb +2 -2
- data/lib/spree/testing_support/factories/order_factory.rb +2 -2
- data/lib/spree/testing_support/factories/payment_factory.rb +5 -0
- data/lib/spree/testing_support/factories/payment_method_factory.rb +8 -0
- data/lib/spree/testing_support/factories/promotion_rule_factory.rb +5 -0
- data/lib/spree/testing_support/factories/refund_factory.rb +9 -1
- data/lib/spree/testing_support/factories/return_authorization_factory.rb +2 -0
- data/lib/spree/testing_support/factories/state_factory.rb +2 -2
- data/lib/spree/testing_support/factories/store_credit_category_factory.rb +9 -0
- data/lib/spree/testing_support/factories/store_credit_event_factory.rb +8 -0
- data/lib/spree/testing_support/factories/store_credit_factory.rb +17 -0
- data/lib/spree/testing_support/factories/store_credit_type_factory.rb +11 -0
- data/lib/spree/testing_support/factories/user_factory.rb +1 -1
- data/lib/spree/testing_support/factories/zone_member_factory.rb +6 -0
- data/lib/spree/testing_support/microdata.rb +189 -0
- data/lib/tasks/core.rake +68 -0
- data/lib/tasks/exchanges.rake +2 -2
- data/spec/fixtures/microdata.html +22 -0
- data/spec/fixtures/microdata_itemref.html +15 -0
- data/spec/fixtures/microdata_no_itemscope.html +20 -0
- data/spec/helpers/base_helper_spec.rb +64 -1
- data/spec/helpers/products_helper_spec.rb +75 -3
- data/spec/lib/i18n_spec.rb +2 -2
- data/spec/lib/search/base_spec.rb +2 -2
- data/spec/lib/spree/core/controller_helpers/auth_spec.rb +4 -2
- data/spec/lib/spree/core/controller_helpers/store_spec.rb +56 -0
- data/spec/lib/spree/core/importer/order_spec.rb +226 -123
- data/spec/lib/spree/core/number_generator_spec.rb +175 -0
- data/spec/lib/spree/core_spec.rb +23 -0
- data/spec/lib/spree/localized_number_spec.rb +10 -0
- data/spec/mailers/order_mailer_spec.rb +11 -13
- data/spec/mailers/shipment_mailer_spec.rb +26 -8
- data/spec/mailers/test_mailer_spec.rb +15 -1
- data/spec/models/option_type_prototype_spec.rb +9 -0
- data/spec/models/spree/ability_spec.rb +6 -13
- data/spec/models/spree/address_spec.rb +1 -1
- data/spec/models/spree/adjustable/adjuster/base_spec.rb +10 -0
- data/spec/models/spree/adjustable/adjuster/promotion_spec.rb +211 -0
- data/spec/models/spree/adjustable/adjuster/tax_spec.rb +86 -0
- data/spec/models/spree/adjustable/adjustments_updater_spec.rb +2 -262
- data/spec/models/spree/adjustment_spec.rb +27 -1
- data/spec/models/spree/app_configuration_spec.rb +5 -2
- data/spec/models/spree/calculator/default_tax_spec.rb +39 -14
- data/spec/models/spree/concerns/user_methods_spec.rb +55 -0
- data/spec/models/spree/concerns/vat_price_calculation_spec.rb +66 -0
- data/spec/models/spree/country_spec.rb +45 -8
- data/spec/models/spree/credit_card_spec.rb +8 -8
- data/spec/models/spree/customer_return_spec.rb +4 -26
- data/spec/models/spree/gateway_spec.rb +7 -0
- data/spec/models/spree/image_spec.rb +3 -0
- data/spec/models/spree/inventory_unit_spec.rb +1 -18
- data/spec/models/spree/line_item_spec.rb +78 -18
- data/spec/models/spree/option_type_spec.rb +2 -2
- data/spec/models/spree/option_value_spec.rb +8 -3
- data/spec/models/spree/order/checkout_spec.rb +49 -39
- data/spec/models/spree/order/currency_updater_spec.rb +3 -3
- data/spec/models/spree/order/finalizing_spec.rb +0 -3
- data/spec/models/spree/order/payment_spec.rb +1 -1
- data/spec/models/spree/order/state_machine_spec.rb +1 -6
- data/spec/models/spree/order/store_credit_spec.rb +423 -0
- data/spec/models/spree/order/updating_spec.rb +2 -2
- data/spec/models/spree/order_contents_spec.rb +42 -1
- data/spec/models/spree/order_inventory_spec.rb +27 -17
- data/spec/models/spree/order_spec.rb +65 -52
- data/spec/models/spree/payment/gateway_options_spec.rb +10 -2
- data/spec/models/spree/payment/store_credit_spec.rb +60 -0
- data/spec/models/spree/payment_method/store_credit_spec.rb +291 -0
- data/spec/models/spree/payment_method_spec.rb +22 -14
- data/spec/models/spree/payment_spec.rb +37 -44
- data/spec/models/spree/price_spec.rb +86 -0
- data/spec/models/spree/product/scopes_spec.rb +35 -0
- data/spec/models/spree/product_option_type_spec.rb +6 -2
- data/spec/models/spree/product_promotion_rule_spec.rb +9 -0
- data/spec/models/spree/product_property_spec.rb +11 -0
- data/spec/models/spree/product_spec.rb +82 -15
- data/spec/models/spree/promotion/actions/create_item_adjustments_spec.rb +1 -1
- data/spec/models/spree/promotion/rules/user_spec.rb +8 -0
- data/spec/models/spree/promotion_action_spec.rb +1 -1
- data/spec/models/spree/promotion_rule_spec.rb +1 -1
- data/spec/models/spree/promotion_rule_taxon_spec.rb +9 -0
- data/spec/models/spree/promotion_rule_user_spec.rb +9 -0
- data/spec/models/spree/promotion_spec.rb +57 -36
- data/spec/models/spree/property_prototype_spec.rb +9 -0
- data/spec/models/spree/prototype_taxon_spec.rb +9 -0
- data/spec/models/spree/refund_reason_spec.rb +7 -0
- data/spec/models/spree/reimbursement_spec.rb +3 -30
- data/spec/models/spree/reimbursement_tax_calculator_spec.rb +17 -5
- data/spec/models/spree/reimbursement_type/store_credit_spec.rb +101 -0
- data/spec/models/spree/return_authorization_reason_spec.rb +7 -0
- data/spec/models/spree/return_authorization_spec.rb +2 -22
- data/spec/models/spree/return_item_spec.rb +50 -1
- data/spec/models/spree/returns_calculator_spec.rb +1 -1
- data/spec/models/spree/role_spec.rb +7 -0
- data/spec/models/spree/shipment_spec.rb +17 -17
- data/spec/models/spree/shipping_calculator_spec.rb +2 -2
- data/spec/models/spree/shipping_category_spec.rb +14 -0
- data/spec/models/spree/shipping_method_spec.rb +9 -2
- data/spec/models/spree/shipping_rate_spec.rb +40 -41
- data/spec/models/spree/state_spec.rb +12 -1
- data/spec/models/spree/stock/content_item_spec.rb +9 -0
- data/spec/models/spree/stock/estimator_spec.rb +56 -8
- data/spec/models/spree/stock/quantifier_spec.rb +61 -32
- data/spec/models/spree/stock_item_spec.rb +19 -1
- data/spec/models/spree/store_credit_event_spec.rb +101 -0
- data/spec/models/spree/store_credit_spec.rb +786 -0
- data/spec/models/spree/store_spec.rb +39 -11
- data/spec/models/spree/tax_category_spec.rb +6 -1
- data/spec/models/spree/tax_rate_spec.rb +204 -44
- data/spec/models/spree/user_spec.rb +105 -38
- data/spec/models/spree/variant_spec.rb +281 -9
- data/spec/models/spree/zone_member_spec.rb +38 -0
- data/spec/models/spree/zone_spec.rb +32 -8
- data/spec/spec_helper.rb +3 -0
- data/spec/support/concerns/{adjustment_source_spec.rb → adjustment_source.rb} +0 -0
- data/spec/support/concerns/{default_price_spec.rb → default_price.rb} +9 -0
- data/spec/validators/db_maximum_length_validator_spec.rb +22 -0
- data/spree_core.gemspec +5 -6
- metadata +99 -36
- data/CHANGELOG.md +0 -4
- data/app/models/concerns/spree/number_generator.rb +0 -39
- data/spec/models/spree/validations/db_maximum_length_validator_spec.rb +0 -24
@@ -0,0 +1,22 @@
|
|
1
|
+
module Spree
|
2
|
+
class StoreCreditCategory < Spree::Base
|
3
|
+
validates_presence_of :name
|
4
|
+
|
5
|
+
GIFT_CARD_CATEGORY_NAME = 'Gift Card'.freeze
|
6
|
+
DEFAULT_NON_EXPIRING_TYPES = [GIFT_CARD_CATEGORY_NAME]
|
7
|
+
|
8
|
+
def non_expiring?
|
9
|
+
non_expiring_category_types.include? name
|
10
|
+
end
|
11
|
+
|
12
|
+
def non_expiring_category_types
|
13
|
+
DEFAULT_NON_EXPIRING_TYPES | Spree::Config[:non_expiring_credit_types]
|
14
|
+
end
|
15
|
+
|
16
|
+
class << self
|
17
|
+
def default_reimbursement_category(_options = {})
|
18
|
+
Spree::StoreCreditCategory.first
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,38 @@
|
|
1
|
+
module Spree
|
2
|
+
class StoreCreditEvent < Spree::Base
|
3
|
+
acts_as_paranoid
|
4
|
+
|
5
|
+
belongs_to :store_credit
|
6
|
+
belongs_to :originator, polymorphic: true
|
7
|
+
|
8
|
+
scope :exposed_events, -> { where.not(action: [Spree::StoreCredit::ELIGIBLE_ACTION, Spree::StoreCredit::AUTHORIZE_ACTION]) }
|
9
|
+
scope :reverse_chronological, -> { order(created_at: :desc) }
|
10
|
+
|
11
|
+
delegate :currency, to: :store_credit
|
12
|
+
|
13
|
+
def display_amount
|
14
|
+
Spree::Money.new(amount, currency: currency)
|
15
|
+
end
|
16
|
+
|
17
|
+
def display_user_total_amount
|
18
|
+
Spree::Money.new(user_total_amount, currency: currency)
|
19
|
+
end
|
20
|
+
|
21
|
+
def display_action
|
22
|
+
case action
|
23
|
+
when Spree::StoreCredit::CAPTURE_ACTION
|
24
|
+
Spree.t('store_credit.captured')
|
25
|
+
when Spree::StoreCredit::AUTHORIZE_ACTION
|
26
|
+
Spree.t('store_credit.authorized')
|
27
|
+
when Spree::StoreCredit::ALLOCATION_ACTION
|
28
|
+
Spree.t('store_credit.allocated')
|
29
|
+
when Spree::StoreCredit::VOID_ACTION, Spree::StoreCredit::CREDIT_ACTION
|
30
|
+
Spree.t('store_credit.credit')
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
def order
|
35
|
+
Spree::Payment.find_by_response_code(authorization_code).try(:order)
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
@@ -1,7 +1,7 @@
|
|
1
1
|
module Spree
|
2
2
|
class TaxCategory < Spree::Base
|
3
3
|
acts_as_paranoid
|
4
|
-
validates :name, presence: true, uniqueness: { scope: :deleted_at, allow_blank: true }
|
4
|
+
validates :name, presence: true, uniqueness: { case_sensitive: false, scope: :deleted_at, allow_blank: true }
|
5
5
|
|
6
6
|
has_many :tax_rates, dependent: :destroy, inverse_of: :tax_category
|
7
7
|
|
@@ -10,13 +10,8 @@ module Spree
|
|
10
10
|
def set_default_category
|
11
11
|
#set existing default tax category to false if this one has been marked as default
|
12
12
|
|
13
|
-
if is_default && tax_category = self.class.where(is_default: true).first
|
14
|
-
|
15
|
-
tax_category.update_columns(
|
16
|
-
is_default: false,
|
17
|
-
updated_at: Time.now,
|
18
|
-
)
|
19
|
-
end
|
13
|
+
if is_default && tax_category = self.class.where(is_default: true).where.not(id: id).first
|
14
|
+
tax_category.update_columns(is_default: false, updated_at: Time.current)
|
20
15
|
end
|
21
16
|
end
|
22
17
|
end
|
@@ -1,13 +1,3 @@
|
|
1
|
-
module Spree
|
2
|
-
class DefaultTaxZoneValidator < ActiveModel::Validator
|
3
|
-
def validate(record)
|
4
|
-
if record.included_in_price
|
5
|
-
record.errors.add(:included_in_price, Spree.t(:included_price_validation)) unless Zone.default_tax
|
6
|
-
end
|
7
|
-
end
|
8
|
-
end
|
9
|
-
end
|
10
|
-
|
11
1
|
module Spree
|
12
2
|
class TaxRate < Spree::Base
|
13
3
|
acts_as_paranoid
|
@@ -15,50 +5,32 @@ module Spree
|
|
15
5
|
include Spree::CalculatedAdjustments
|
16
6
|
include Spree::AdjustmentSource
|
17
7
|
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
validates_with DefaultTaxZoneValidator
|
24
|
-
|
25
|
-
scope :by_zone, ->(zone) { where(zone_id: zone) }
|
8
|
+
with_options inverse_of: :tax_rates do
|
9
|
+
belongs_to :zone, class_name: "Spree::Zone"
|
10
|
+
belongs_to :tax_category,
|
11
|
+
class_name: "Spree::TaxCategory"
|
12
|
+
end
|
26
13
|
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
merge(Spree::Zone.potential_matching_zones(zone)).
|
31
|
-
order("spree_zones.default_tax DESC")
|
14
|
+
with_options presence: true do
|
15
|
+
validates :amount, numericality: { allow_nil: true }
|
16
|
+
validates :tax_category
|
32
17
|
end
|
33
18
|
|
34
|
-
|
19
|
+
scope :by_zone, -> (zone) { where(zone_id: zone.id) }
|
20
|
+
scope :potential_rates_for_zone,
|
21
|
+
-> (zone) do
|
22
|
+
where(zone_id: Spree::Zone.potential_matching_zones(zone).pluck(:id))
|
23
|
+
end
|
24
|
+
scope :for_default_zone,
|
25
|
+
-> { potential_rates_for_zone(Spree::Zone.default_tax) }
|
26
|
+
scope :for_tax_category,
|
27
|
+
-> (category) { where(tax_category_id: category.try(:id)) }
|
28
|
+
scope :included_in_price, -> { where(included_in_price: true) }
|
29
|
+
|
30
|
+
# Gets the array of TaxRates appropriate for the specified tax zone
|
35
31
|
def self.match(order_tax_zone)
|
36
32
|
return [] unless order_tax_zone
|
37
|
-
|
38
|
-
potential_rates = potential_rates_for_zone(order_tax_zone)
|
39
|
-
rates = potential_rates.includes(zone: { zone_members: :zoneable }).load.select do |rate|
|
40
|
-
# Why "potentially"?
|
41
|
-
# Go see the documentation for that method.
|
42
|
-
rate.potentially_applicable?(order_tax_zone)
|
43
|
-
end
|
44
|
-
|
45
|
-
# Imagine with me this scenario:
|
46
|
-
# You are living in Spain and you have a store which ships to France.
|
47
|
-
# Spain is therefore your default tax rate.
|
48
|
-
# When you ship to Spain, you want the Spanish rate to apply.
|
49
|
-
# When you ship to France, you want the French rate to apply.
|
50
|
-
#
|
51
|
-
# Normally, Spree would notice that you have two potentially applicable
|
52
|
-
# tax rates for one particular item.
|
53
|
-
# When you ship to Spain, only the Spanish one will apply.
|
54
|
-
# When you ship to France, you'll see a Spanish refund AND a French tax.
|
55
|
-
# This little bit of code at the end stops the Spanish refund from appearing.
|
56
|
-
#
|
57
|
-
# For further discussion, see #4397 and #4327.
|
58
|
-
rates.delete_if do |rate|
|
59
|
-
rate.included_in_price? &&
|
60
|
-
(rates - [rate]).map(&:tax_category).include?(rate.tax_category)
|
61
|
-
end
|
33
|
+
potential_rates_for_zone(order_tax_zone)
|
62
34
|
end
|
63
35
|
|
64
36
|
# Pre-tax amounts must be stored so that we can calculate
|
@@ -66,9 +38,9 @@ module Spree
|
|
66
38
|
# https://github.com/spree/spree/issues/4318#issuecomment-34723428
|
67
39
|
def self.store_pre_tax_amount(item, rates)
|
68
40
|
pre_tax_amount = case item
|
69
|
-
|
70
|
-
|
71
|
-
|
41
|
+
when Spree::LineItem then item.discounted_amount
|
42
|
+
when Spree::Shipment then item.discounted_cost
|
43
|
+
end
|
72
44
|
|
73
45
|
included_rates = rates.select(&:included_in_price)
|
74
46
|
if included_rates.any?
|
@@ -78,99 +50,61 @@ module Spree
|
|
78
50
|
item.update_column(:pre_tax_amount, pre_tax_amount)
|
79
51
|
end
|
80
52
|
|
81
|
-
#
|
53
|
+
# Deletes all tax adjustments, then applies all applicable rates
|
54
|
+
# to relevant items.
|
82
55
|
def self.adjust(order, items)
|
83
56
|
rates = match(order.tax_zone)
|
84
57
|
tax_categories = rates.map(&:tax_category)
|
85
|
-
|
86
|
-
|
58
|
+
|
59
|
+
# using destroy_all to ensure adjustment destroy callback fires.
|
60
|
+
Spree::Adjustment.where(adjustable: items).tax.destroy_all
|
61
|
+
|
62
|
+
relevant_items = items.select do |item|
|
63
|
+
tax_categories.include?(item.tax_category)
|
64
|
+
end
|
65
|
+
|
87
66
|
relevant_items.each do |item|
|
88
|
-
relevant_rates = rates.select
|
67
|
+
relevant_rates = rates.select do |rate|
|
68
|
+
rate.tax_category == item.tax_category
|
69
|
+
end
|
89
70
|
store_pre_tax_amount(item, relevant_rates)
|
90
71
|
relevant_rates.each do |rate|
|
91
72
|
rate.adjust(order, item)
|
92
73
|
end
|
93
74
|
end
|
94
|
-
non_relevant_items.each do |item|
|
95
|
-
if item.adjustments.tax.present?
|
96
|
-
item.adjustments.tax.destroy_all # using destroy_all to ensure adjustment destroy callback fires.
|
97
|
-
item.update_columns pre_tax_amount: 0
|
98
|
-
end
|
99
|
-
end
|
100
75
|
end
|
101
76
|
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
# For instance:
|
109
|
-
#
|
110
|
-
# Zones:
|
111
|
-
# - Spain (default tax zone)
|
112
|
-
# - France
|
113
|
-
#
|
114
|
-
# Tax rates: (note: amounts below do not actually reflect real VAT rates)
|
115
|
-
# 21% inclusive - "Clothing" - Spain
|
116
|
-
# 18% inclusive - "Clothing" - France
|
117
|
-
# 10% inclusive - "Food" - Spain
|
118
|
-
# 8% inclusive - "Food" - France
|
119
|
-
# 5% inclusive - "Hotels" - Spain
|
120
|
-
# 2% inclusive - "Hotels" - France
|
121
|
-
#
|
122
|
-
# Order has:
|
123
|
-
# Line Item #1 - Tax Category: Clothing
|
124
|
-
# Line Item #2 - Tax Category: Food
|
125
|
-
#
|
126
|
-
# Tax rates that should be selected:
|
127
|
-
#
|
128
|
-
# 21% inclusive - "Clothing" - Spain
|
129
|
-
# 10% inclusive - "Food" - Spain
|
130
|
-
#
|
131
|
-
# If the order's address changes to one in France, then the tax will be recalculated:
|
132
|
-
#
|
133
|
-
# 18% inclusive - "Clothing" - France
|
134
|
-
# 8% inclusive - "Food" - France
|
135
|
-
#
|
136
|
-
# Note here that the "Hotels" tax rates will not be used at all.
|
137
|
-
# This is because there are no items which have the tax category of "Hotels".
|
138
|
-
#
|
139
|
-
# Under no circumstances should negative adjustments be applied for the Spanish tax rates.
|
140
|
-
#
|
141
|
-
# Those rates should never come into play at all and only the French rates should apply.
|
142
|
-
def potentially_applicable?(order_tax_zone)
|
143
|
-
# If the rate's zone matches the order's tax zone, then it's applicable.
|
144
|
-
self.zone == order_tax_zone ||
|
145
|
-
# If the rate's zone *contains* the order's tax zone, then it's applicable.
|
146
|
-
self.zone.contains?(order_tax_zone) ||
|
147
|
-
# 1) The rate's zone is the default zone, then it's always applicable.
|
148
|
-
(self.included_in_price? && self.zone.default_tax)
|
77
|
+
def self.included_tax_amount_for(options)
|
78
|
+
return 0 unless options[:tax_zone] && options[:tax_category]
|
79
|
+
potential_rates_for_zone(options[:tax_zone]).
|
80
|
+
included_in_price.
|
81
|
+
for_tax_category(options[:tax_category]).
|
82
|
+
pluck(:amount).sum
|
149
83
|
end
|
150
84
|
|
151
85
|
def adjust(order, item)
|
152
|
-
|
153
|
-
create_adjustment(order, item, included)
|
86
|
+
create_adjustment(order, item, included_in_price)
|
154
87
|
end
|
155
88
|
|
156
89
|
def compute_amount(item)
|
157
|
-
|
158
|
-
compute(item) * (refund ? -1 : 1)
|
90
|
+
compute(item)
|
159
91
|
end
|
160
92
|
|
161
93
|
private
|
162
94
|
|
163
|
-
def
|
164
|
-
|
95
|
+
def label
|
96
|
+
Spree.t included_in_price? ? :including_tax : :excluding_tax,
|
97
|
+
scope: "adjustment_labels.tax_rates",
|
98
|
+
name: name.presence || tax_category.name,
|
99
|
+
amount: amount_for_label
|
165
100
|
end
|
166
101
|
|
167
|
-
def
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
label
|
102
|
+
def amount_for_label
|
103
|
+
return "" unless show_rate_in_label?
|
104
|
+
" " + ActiveSupport::NumberHelper::NumberToPercentageConverter.convert(
|
105
|
+
amount * 100,
|
106
|
+
locale: I18n.locale
|
107
|
+
)
|
174
108
|
end
|
175
109
|
end
|
176
110
|
end
|
data/app/models/spree/taxon.rb
CHANGED
@@ -13,12 +13,18 @@ module Spree
|
|
13
13
|
has_many :classifications, -> { order(:position) }, dependent: :delete_all, inverse_of: :taxon
|
14
14
|
has_many :products, through: :classifications
|
15
15
|
|
16
|
-
|
16
|
+
has_many :prototype_taxons, class_name: 'Spree::PrototypeTaxon'
|
17
|
+
has_many :prototypes, through: :prototype_taxons, class_name: 'Spree::Prototype'
|
18
|
+
|
19
|
+
has_many :promotion_rule_taxons, class_name: 'Spree::PromotionRuleTaxon'
|
20
|
+
has_many :promotion_rules, through: :promotion_rule_taxons, class_name: 'Spree::PromotionRule'
|
17
21
|
|
18
22
|
validates :name, presence: true
|
19
|
-
|
20
|
-
|
21
|
-
|
23
|
+
with_options length: { maximum: 255 }, allow_blank: true do
|
24
|
+
validates :meta_keywords
|
25
|
+
validates :meta_description
|
26
|
+
validates :meta_title
|
27
|
+
end
|
22
28
|
|
23
29
|
after_save :touch_ancestors_and_taxonomy
|
24
30
|
after_touch :touch_ancestors_and_taxonomy
|
@@ -88,7 +94,7 @@ module Spree
|
|
88
94
|
|
89
95
|
def touch_ancestors_and_taxonomy
|
90
96
|
# Touches all ancestors at once to avoid recursive taxonomy touch, and reduce queries.
|
91
|
-
|
97
|
+
ancestors.update_all(updated_at: Time.current)
|
92
98
|
# Have taxonomy touch happen in #touch_ancestors_and_taxonomy rather than association option in order for imports to override.
|
93
99
|
taxonomy.try!(:touch)
|
94
100
|
end
|
data/app/models/spree/tracker.rb
CHANGED
@@ -1,8 +1,19 @@
|
|
1
1
|
module Spree
|
2
2
|
class Tracker < Spree::Base
|
3
|
+
after_commit :clear_cache
|
4
|
+
|
5
|
+
validates :analytics_id, presence: true, uniqueness: { allow_blank: true }
|
6
|
+
|
3
7
|
def self.current
|
4
|
-
tracker =
|
8
|
+
tracker = Rails.cache.fetch("current_tracker") do
|
9
|
+
where(active: true).first
|
10
|
+
end
|
5
11
|
tracker.analytics_id.present? ? tracker : nil if tracker
|
6
12
|
end
|
13
|
+
|
14
|
+
|
15
|
+
def clear_cache
|
16
|
+
Rails.cache.delete("current_tracker")
|
17
|
+
end
|
7
18
|
end
|
8
19
|
end
|
@@ -11,9 +11,10 @@ module Spree
|
|
11
11
|
end
|
12
12
|
|
13
13
|
def validate(record)
|
14
|
+
warn "`Spree::Validations::DbMaximumLengthValidator` is deprecated. Use `DbMaximumLengthValidator` instead."
|
14
15
|
limit = record.class.columns_hash[@field].limit
|
15
16
|
value = record[@field.to_sym]
|
16
|
-
if value && limit && value.to_s.length > limit
|
17
|
+
if value && limit && value.to_s.length > limit
|
17
18
|
record.errors.add(@field.to_sym, :too_long, count: limit)
|
18
19
|
end
|
19
20
|
end
|
data/app/models/spree/variant.rb
CHANGED
@@ -12,15 +12,22 @@ module Spree
|
|
12
12
|
:shipping_category_id, :meta_description, :meta_keywords,
|
13
13
|
:shipping_category
|
14
14
|
|
15
|
-
|
16
|
-
|
15
|
+
with_options inverse_of: :variant do
|
16
|
+
has_many :inventory_units
|
17
|
+
has_many :stock_items, dependent: :destroy
|
18
|
+
end
|
19
|
+
|
20
|
+
has_many :line_items, dependent: :restrict_with_error
|
21
|
+
|
17
22
|
has_many :orders, through: :line_items
|
23
|
+
with_options through: :stock_items do
|
24
|
+
has_many :stock_locations
|
25
|
+
has_many :stock_movements
|
26
|
+
end
|
18
27
|
|
19
|
-
has_many :
|
20
|
-
has_many :
|
21
|
-
has_many :stock_movements, through: :stock_items
|
28
|
+
has_many :option_value_variants, class_name: 'Spree::OptionValueVariant'
|
29
|
+
has_many :option_values, through: :option_value_variants, class_name: 'Spree::OptionValue'
|
22
30
|
|
23
|
-
has_and_belongs_to_many :option_values, join_table: :spree_option_values_variants
|
24
31
|
has_many :images, -> { order(:position) }, as: :viewable, dependent: :destroy, class_name: "Spree::Image"
|
25
32
|
|
26
33
|
has_many :prices,
|
@@ -32,9 +39,13 @@ module Spree
|
|
32
39
|
|
33
40
|
validate :check_price
|
34
41
|
|
35
|
-
validates :
|
36
|
-
|
37
|
-
|
42
|
+
validates :option_values, presence: true, unless: :is_master?
|
43
|
+
|
44
|
+
with_options numericality: { greater_than_or_equal_to: 0, allow_nil: true } do
|
45
|
+
validates :cost_price
|
46
|
+
validates :price
|
47
|
+
end
|
48
|
+
validates :sku, uniqueness: { conditions: -> { where(deleted_at: nil) } }, allow_blank: true
|
38
49
|
|
39
50
|
after_create :create_stock_items
|
40
51
|
after_create :set_master_out_of_stock, unless: :is_master?
|
@@ -43,6 +54,23 @@ module Spree
|
|
43
54
|
|
44
55
|
scope :in_stock, -> { joins(:stock_items).where('count_on_hand > ? OR track_inventory = ?', 0, false) }
|
45
56
|
|
57
|
+
scope :not_discontinued, -> do
|
58
|
+
variant_table_name = Variant.quoted_table_name
|
59
|
+
where("#{variant_table_name}.discontinue_on IS NULL OR #{variant_table_name}.discontinue_on >= ?", Time.current)
|
60
|
+
end
|
61
|
+
|
62
|
+
scope :not_deleted, -> { where("#{Variant.quoted_table_name}.deleted_at IS NULL") }
|
63
|
+
|
64
|
+
scope :for_currency_and_available_price_amount, -> (currency) do
|
65
|
+
currency ||= Spree::Config[:currency]
|
66
|
+
joins(:prices).where("spree_prices.currency = ?", currency).where("spree_prices.amount IS NOT NULL").uniq
|
67
|
+
end
|
68
|
+
|
69
|
+
scope :active, -> (currency = nil) do
|
70
|
+
not_discontinued.not_deleted.
|
71
|
+
for_currency_and_available_price_amount(currency)
|
72
|
+
end
|
73
|
+
|
46
74
|
LOCALIZED_NUMBERS = %w(cost_price weight depth width height)
|
47
75
|
|
48
76
|
LOCALIZED_NUMBERS.each do |m|
|
@@ -54,8 +82,8 @@ module Spree
|
|
54
82
|
self.whitelisted_ransackable_associations = %w[option_values product prices default_price]
|
55
83
|
self.whitelisted_ransackable_attributes = %w[weight sku]
|
56
84
|
|
57
|
-
def
|
58
|
-
|
85
|
+
def available?
|
86
|
+
!discontinued? && product.available?
|
59
87
|
end
|
60
88
|
|
61
89
|
def self.having_orders
|
@@ -75,10 +103,6 @@ module Spree
|
|
75
103
|
inventory_units.with_state('backordered').size
|
76
104
|
end
|
77
105
|
|
78
|
-
def is_backorderable?
|
79
|
-
Spree::Stock::Quantifier.new(self).backorderable?
|
80
|
-
end
|
81
|
-
|
82
106
|
def options_text
|
83
107
|
values = self.option_values.sort do |a, b|
|
84
108
|
a.option_type.position <=> b.option_type.position
|
@@ -155,7 +179,7 @@ module Spree
|
|
155
179
|
end
|
156
180
|
|
157
181
|
def price_in(currency)
|
158
|
-
prices.detect { |price| price.currency == currency } ||
|
182
|
+
prices.detect { |price| price.currency == currency } || prices.build(currency: currency)
|
159
183
|
end
|
160
184
|
|
161
185
|
def amount_in(currency)
|
@@ -202,13 +226,9 @@ module Spree
|
|
202
226
|
end
|
203
227
|
end
|
204
228
|
|
205
|
-
|
206
|
-
Spree::Stock::Quantifier.new(self).can_supply?(quantity)
|
207
|
-
end
|
229
|
+
delegate :total_on_hand, :can_supply?, :backorderable?, to: :quantifier
|
208
230
|
|
209
|
-
|
210
|
-
Spree::Stock::Quantifier.new(self).total_on_hand
|
211
|
-
end
|
231
|
+
alias is_backorderable? backorderable?
|
212
232
|
|
213
233
|
# Shortcut method to determine if inventory tracking is enabled for this variant
|
214
234
|
# This considers both variant tracking flag and site-wide inventory tracking settings
|
@@ -228,43 +248,55 @@ module Spree
|
|
228
248
|
(width || 0) + (height || 0) + (depth || 0)
|
229
249
|
end
|
230
250
|
|
251
|
+
def discontinue!
|
252
|
+
update_attribute(:discontinue_on, Time.current)
|
253
|
+
end
|
254
|
+
|
255
|
+
def discontinued?
|
256
|
+
!!discontinue_on && discontinue_on <= Time.current
|
257
|
+
end
|
258
|
+
|
231
259
|
private
|
232
260
|
|
233
|
-
|
234
|
-
|
235
|
-
|
236
|
-
product.master.stock_items.each { |item| item.reduce_count_on_hand_to_zero }
|
237
|
-
end
|
238
|
-
end
|
261
|
+
def quantifier
|
262
|
+
Spree::Stock::Quantifier.new(self)
|
263
|
+
end
|
239
264
|
|
240
|
-
|
241
|
-
|
242
|
-
|
243
|
-
|
244
|
-
raise 'Must supply price for variant or master.price for product.' if self == product.master
|
245
|
-
self.price = product.master.price
|
246
|
-
end
|
247
|
-
if currency.nil?
|
248
|
-
self.currency = Spree::Config[:currency]
|
249
|
-
end
|
265
|
+
def set_master_out_of_stock
|
266
|
+
if product.master && product.master.in_stock?
|
267
|
+
product.master.stock_items.update_all(backorderable: false)
|
268
|
+
product.master.stock_items.each(&:reduce_count_on_hand_to_zero)
|
250
269
|
end
|
270
|
+
end
|
251
271
|
|
252
|
-
|
253
|
-
|
272
|
+
# Ensures a new variant takes the product master price when price is not supplied
|
273
|
+
def check_price
|
274
|
+
if price.nil? && Spree::Config[:require_master_price]
|
275
|
+
raise 'No master variant found to infer price' unless product && product.master
|
276
|
+
raise 'Must supply price for variant or master.price for product.' if self == product.master
|
277
|
+
self.price = product.master.price
|
254
278
|
end
|
255
|
-
|
256
|
-
|
257
|
-
StockLocation.where(propagate_all_variants: true).each do |stock_location|
|
258
|
-
stock_location.propagate_variant(self)
|
259
|
-
end
|
279
|
+
if price.present? && currency.nil?
|
280
|
+
self.currency = Spree::Config[:currency]
|
260
281
|
end
|
282
|
+
end
|
261
283
|
|
262
|
-
|
263
|
-
|
264
|
-
|
284
|
+
def set_cost_currency
|
285
|
+
self.cost_currency = Spree::Config[:currency] if cost_currency.blank?
|
286
|
+
end
|
265
287
|
|
266
|
-
|
267
|
-
|
288
|
+
def create_stock_items
|
289
|
+
StockLocation.where(propagate_all_variants: true).each do |stock_location|
|
290
|
+
stock_location.propagate_variant(self)
|
268
291
|
end
|
292
|
+
end
|
293
|
+
|
294
|
+
def in_stock_cache_key
|
295
|
+
"variant-#{id}-in_stock"
|
296
|
+
end
|
297
|
+
|
298
|
+
def clear_in_stock_cache
|
299
|
+
Rails.cache.delete(in_stock_cache_key)
|
300
|
+
end
|
269
301
|
end
|
270
302
|
end
|