solidus_core 4.1.6 → 4.5.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/app/assets/images/logo/solidus.svg +18 -1
- data/app/assets/images/logo/solidus_logo.png +0 -0
- data/app/helpers/spree/base_helper.rb +2 -2
- data/app/helpers/spree/core/controller_helpers/auth.rb +66 -0
- data/app/helpers/spree/core/controller_helpers/common.rb +82 -0
- data/app/helpers/spree/core/controller_helpers/order.rb +86 -0
- data/app/helpers/spree/core/controller_helpers/payment_parameters.rb +165 -0
- data/app/helpers/spree/core/controller_helpers/pricing.rb +19 -0
- data/app/helpers/spree/core/controller_helpers/search.rb +16 -0
- data/app/helpers/spree/core/controller_helpers/store.rb +19 -0
- data/app/helpers/spree/core/controller_helpers/strong_parameters.rb +74 -0
- data/app/mailers/spree/base_mailer.rb +1 -1
- data/app/mailers/spree/carton_mailer.rb +1 -1
- data/app/mailers/spree/order_mailer.rb +3 -3
- data/app/mailers/spree/reimbursement_mailer.rb +1 -1
- data/app/models/concerns/spree/active_storage_adapter/normalization.rb +1 -1
- data/app/models/concerns/spree/active_storage_adapter.rb +1 -1
- data/app/models/concerns/spree/display_money.rb +1 -1
- data/app/models/concerns/spree/metadata.rb +64 -0
- data/app/models/concerns/spree/named_type.rb +2 -0
- data/app/models/concerns/spree/ordered_property_value_list.rb +2 -2
- data/app/models/concerns/spree/user_address_book.rb +8 -9
- data/app/models/concerns/spree/user_methods.rb +3 -3
- data/app/models/spree/address.rb +10 -8
- data/app/models/spree/adjustment.rb +15 -65
- data/app/models/spree/adjustment_reason.rb +2 -0
- data/app/models/spree/calculator/returns/default_refund_amount.rb +1 -1
- data/app/models/spree/carton.rb +2 -2
- data/app/models/spree/core/state_machines/inventory_unit.rb +42 -0
- data/app/models/spree/core/state_machines/order/class_methods.rb +217 -0
- data/app/models/spree/core/state_machines/order.rb +42 -0
- data/app/models/spree/core/state_machines/payment.rb +61 -0
- data/app/models/spree/core/state_machines/reimbursement.rb +33 -0
- data/app/models/spree/core/state_machines/return_authorization.rb +32 -0
- data/app/models/spree/core/state_machines/return_item/acceptance_status.rb +51 -0
- data/app/models/spree/core/state_machines/return_item/reception_status.rb +42 -0
- data/app/models/spree/core/state_machines/shipment.rb +58 -0
- data/app/models/spree/credit_card.rb +12 -9
- data/app/models/spree/customer_return.rb +2 -0
- data/app/models/spree/deprecated_configurable_class.rb +40 -0
- data/app/models/spree/fulfilment_changer.rb +4 -4
- data/app/models/spree/inventory_unit.rb +2 -2
- data/app/models/spree/item_total.rb +28 -0
- data/app/models/spree/legacy_user.rb +1 -0
- data/app/models/spree/line_item.rb +22 -4
- data/app/models/spree/money.rb +120 -0
- data/app/models/spree/null_promotion_adjuster.rb +13 -0
- data/app/models/spree/null_promotion_advertiser.rb +9 -0
- data/app/models/spree/null_promotion_finder.rb +9 -0
- data/app/models/spree/null_promotion_handler.rb +44 -0
- data/app/models/spree/option_value.rb +1 -0
- data/app/models/spree/order.rb +54 -36
- data/app/models/spree/order_cancellations.rb +8 -8
- data/app/models/spree/order_inventory.rb +6 -4
- data/app/models/spree/order_merger.rb +1 -1
- data/app/models/spree/order_mutex.rb +2 -2
- data/app/models/spree/order_shipping.rb +9 -9
- data/app/models/spree/order_taxation.rb +1 -0
- data/app/models/spree/order_updater.rb +19 -40
- data/app/models/spree/payment/processing.rb +2 -2
- data/app/models/spree/payment.rb +3 -2
- data/app/models/spree/payment_create.rb +1 -1
- data/app/models/spree/payment_method/bogus_credit_card.rb +14 -9
- data/app/models/spree/payment_method/simple_bogus_credit_card.rb +12 -6
- data/app/models/spree/payment_method/store_credit.rb +1 -1
- data/app/models/spree/payment_method.rb +3 -1
- data/app/models/spree/payment_source.rb +5 -1
- data/app/models/spree/permission_set.rb +11 -0
- data/app/models/spree/permission_sets/base.rb +45 -0
- data/app/models/spree/permission_sets/configuration_display.rb +53 -0
- data/app/models/spree/permission_sets/configuration_management.rb +52 -0
- data/app/models/spree/permission_sets/dashboard_display.rb +28 -0
- data/app/models/spree/permission_sets/default_customer.rb +83 -0
- data/app/models/spree/permission_sets/order_display.rb +50 -0
- data/app/models/spree/permission_sets/order_management.rb +50 -0
- data/app/models/spree/permission_sets/product_display.rb +43 -0
- data/app/models/spree/permission_sets/product_management.rb +47 -0
- data/app/models/spree/permission_sets/restricted_stock_display.rb +33 -0
- data/app/models/spree/permission_sets/restricted_stock_management.rb +33 -0
- data/app/models/spree/permission_sets/stock_display.rb +26 -0
- data/app/models/spree/permission_sets/stock_management.rb +26 -0
- data/app/models/spree/permission_sets/super_user.rb +26 -0
- data/app/models/spree/permission_sets/user_display.rb +27 -0
- data/app/models/spree/permission_sets/user_management.rb +44 -0
- data/app/models/spree/preference.rb +1 -1
- data/app/models/spree/product.rb +13 -9
- data/app/models/spree/refund.rb +3 -1
- data/app/models/spree/refund_reason.rb +6 -1
- data/app/models/spree/reimbursement.rb +4 -4
- data/app/models/spree/reimbursement_performer.rb +3 -3
- data/app/models/spree/reimbursement_tax_calculator.rb +2 -2
- data/app/models/spree/reimbursement_type/credit.rb +1 -1
- data/app/models/spree/reimbursement_type/reimbursement_helpers.rb +6 -6
- data/app/models/spree/reimbursement_type/store_credit.rb +1 -1
- data/app/models/spree/reimbursement_type.rb +6 -1
- data/app/models/spree/return_authorization.rb +2 -0
- data/app/models/spree/return_item.rb +10 -10
- data/app/models/spree/return_reason.rb +6 -1
- data/app/models/spree/role.rb +3 -1
- data/app/models/spree/role_permission.rb +8 -0
- data/app/models/spree/shipment.rb +9 -7
- data/app/models/spree/shipping_category.rb +2 -0
- data/app/models/spree/shipping_rate.rb +2 -3
- data/app/models/spree/{order_contents.rb → simple_order_contents.rb} +8 -12
- data/app/models/spree/stock/estimator.rb +1 -1
- data/app/models/spree/stock/inventory_unit_builder.rb +1 -1
- data/app/models/spree/stock/package.rb +2 -2
- data/app/models/spree/stock/simple_coordinator.rb +36 -22
- data/app/models/spree/stock_item.rb +1 -0
- data/app/models/spree/stock_location.rb +5 -5
- data/app/models/spree/store.rb +3 -1
- data/app/models/spree/store_credit.rb +17 -15
- data/app/models/spree/store_credit_event.rb +4 -3
- data/app/models/spree/store_credit_reason.rb +6 -1
- data/app/models/spree/store_selector/by_server_name.rb +1 -1
- data/app/models/spree/tax/tax_helpers.rb +12 -1
- data/app/models/spree/tax_calculator/default.rb +3 -3
- data/app/models/spree/tax_calculator/shipping_rate.rb +1 -1
- data/app/models/spree/tax_category.rb +3 -1
- data/app/models/spree/tax_rate.rb +7 -3
- data/app/models/spree/taxon.rb +22 -2
- data/app/models/spree/taxon_brand_selector.rb +22 -0
- data/app/models/spree/taxonomy.rb +2 -2
- data/app/models/spree/unauthorized_redirect_handler.rb +24 -0
- data/app/models/spree/unit_cancel.rb +1 -2
- data/app/models/spree/user_address.rb +9 -3
- data/app/models/spree/variant/scopes.rb +4 -0
- data/app/models/spree/variant/vat_price_generator.rb +1 -1
- data/app/models/spree/variant.rb +21 -8
- data/app/models/spree/wallet.rb +2 -2
- data/app/views/spree/order_mailer/cancel_email.html.erb +1 -1
- data/app/views/spree/order_mailer/cancel_email.text.erb +1 -1
- data/app/views/spree/order_mailer/confirm_email.html.erb +5 -5
- data/app/views/spree/order_mailer/confirm_email.text.erb +5 -5
- data/app/views/spree/order_mailer/inventory_cancellation_email.html.erb +0 -1
- data/config/locales/en.yml +33 -90
- data/db/default/spree/permission_sets.rb +10 -0
- data/db/default/spree/return_reasons.rb +3 -1
- data/db/default/spree/states.rb +1 -1
- data/db/migrate/20160101010000_solidus_one_four.rb +0 -127
- data/db/migrate/20220419170826_remove_archived_user_addresses.rb +12 -0
- data/db/migrate/20230427095534_drop_deprecated_address_id_from_shipments.rb +1 -1
- data/db/migrate/20240821173254_create_spree_permission_sets_in_core.rb +9 -0
- data/db/migrate/20240821173341_create_spree_roles_permissions_in_core.rb +9 -0
- data/db/migrate/20240821173641_add_description_to_spree_roles.rb +5 -0
- data/db/migrate/20240904152041_add_privilege_and_category_to_spree_permission_sets.rb +6 -0
- data/db/migrate/20250129061658_add_metadata_to_spree_resources.rb +28 -0
- data/db/migrate/20250201172950_add_gtin_and_condition_to_spree_variant.rb +6 -0
- data/db/migrate/20250207104016_add_primary_taxon_to_products.rb +7 -0
- data/db/migrate/20250221152004_add_metadata_to_users.rb +13 -0
- data/db/seeds.rb +1 -0
- data/lib/generators/solidus/install/app_templates/authentication/custom.rb +0 -5
- data/lib/generators/solidus/install/app_templates/frontend/starter.rb +1 -1
- data/lib/generators/solidus/install/install_generator.rb +64 -15
- data/lib/generators/solidus/install/templates/config/initializers/spree.rb.tt +15 -12
- data/lib/generators/solidus/update/update_generator.rb +1 -1
- data/lib/generators/spree/dummy/dummy_generator.rb +1 -1
- data/lib/generators/spree/dummy/templates/rails/application.rb.tt +1 -1
- data/lib/generators/spree/dummy/templates/rails/database.yml +41 -93
- data/lib/generators/spree/dummy/templates/rails/manifest.js +3 -0
- data/lib/generators/spree/dummy/templates/rails/test.rb +6 -1
- data/lib/spree/app_configuration.rb +102 -67
- data/lib/spree/core/class_constantizer.rb +2 -2
- data/lib/spree/core/controller_helpers/auth.rb +5 -61
- data/lib/spree/core/controller_helpers/common.rb +5 -80
- data/lib/spree/core/controller_helpers/order.rb +5 -86
- data/lib/spree/core/controller_helpers/payment_parameters.rb +5 -163
- data/lib/spree/core/controller_helpers/pricing.rb +5 -17
- data/lib/spree/core/controller_helpers/search.rb +5 -14
- data/lib/spree/core/controller_helpers/store.rb +5 -17
- data/lib/spree/core/controller_helpers/strong_parameters.rb +5 -71
- data/lib/spree/core/engine.rb +13 -5
- data/lib/spree/core/environment/calculators.rb +35 -3
- data/lib/spree/core/environment/promotions.rb +25 -4
- data/lib/spree/core/environment_extension.rb +16 -2
- data/lib/spree/core/importer/order.rb +5 -5
- data/lib/spree/core/importer/product.rb +3 -3
- data/lib/spree/core/nested_class_set.rb +28 -0
- data/lib/spree/core/null_promotion_configuration.rb +84 -0
- data/lib/spree/core/product_filters.rb +1 -1
- data/lib/spree/core/search/variant.rb +2 -2
- data/lib/spree/core/state_machines/inventory_unit.rb +5 -40
- data/lib/spree/core/state_machines/order.rb +5 -251
- data/lib/spree/core/state_machines/payment.rb +5 -59
- data/lib/spree/core/state_machines/reimbursement.rb +5 -31
- data/lib/spree/core/state_machines/return_authorization.rb +5 -30
- data/lib/spree/core/state_machines/return_item/acceptance_status.rb +5 -49
- data/lib/spree/core/state_machines/return_item/reception_status.rb +5 -40
- data/lib/spree/core/state_machines/shipment.rb +5 -56
- data/lib/spree/core/state_machines.rb +48 -81
- data/lib/spree/core/validators/email.rb +1 -1
- data/lib/spree/core/version.rb +2 -2
- data/lib/spree/core.rb +2 -14
- data/lib/spree/deprecator.rb +9 -0
- data/lib/spree/i18n.rb +1 -1
- data/lib/spree/mailer_previews/carton_preview.rb +1 -1
- data/lib/spree/money.rb +5 -118
- data/lib/spree/permission_sets/base.rb +5 -30
- data/lib/spree/permission_sets/configuration_display.rb +5 -41
- data/lib/spree/permission_sets/configuration_management.rb +5 -40
- data/lib/spree/permission_sets/dashboard_display.rb +5 -14
- data/lib/spree/permission_sets/default_customer.rb +5 -71
- data/lib/spree/permission_sets/order_display.rb +5 -38
- data/lib/spree/permission_sets/order_management.rb +5 -38
- data/lib/spree/permission_sets/product_display.rb +5 -31
- data/lib/spree/permission_sets/product_management.rb +5 -35
- data/lib/spree/permission_sets/restricted_stock_display.rb +5 -21
- data/lib/spree/permission_sets/restricted_stock_management.rb +5 -21
- data/lib/spree/permission_sets/stock_display.rb +5 -14
- data/lib/spree/permission_sets/stock_management.rb +5 -14
- data/lib/spree/permission_sets/super_user.rb +5 -14
- data/lib/spree/permission_sets/user_display.rb +5 -15
- data/lib/spree/permission_sets/user_management.rb +5 -32
- data/lib/spree/permission_sets.rb +5 -18
- data/lib/spree/permitted_attributes.rb +21 -9
- data/lib/spree/preferences/configuration.rb +4 -0
- data/lib/spree/preferences/persistable.rb +7 -1
- data/lib/spree/preferences/store.rb +2 -2
- data/lib/spree/testing_support/capybara_ext.rb +11 -1
- data/lib/spree/testing_support/common_rake.rb +1 -0
- data/lib/spree/testing_support/dummy_ability.rb +7 -0
- data/lib/spree/testing_support/dummy_app/assets/stylesheets/solidus_admin/tailwind.css +1 -0
- data/lib/spree/testing_support/dummy_app/database.yml +37 -46
- data/lib/spree/testing_support/dummy_app/migrations.rb +8 -15
- data/lib/spree/testing_support/dummy_app/rake_tasks.rb +3 -6
- data/lib/spree/testing_support/dummy_app.rb +33 -19
- data/lib/spree/testing_support/factories/adjustment_factory.rb +1 -2
- data/lib/spree/testing_support/factories/calculator_factory.rb +1 -5
- data/lib/spree/testing_support/factories/carton_factory.rb +2 -2
- data/lib/spree/testing_support/factories/customer_return_factory.rb +2 -2
- data/lib/spree/testing_support/factories/inventory_unit_factory.rb +3 -3
- data/lib/spree/testing_support/factories/order_factory.rb +13 -24
- data/lib/spree/testing_support/factories/product_factory.rb +1 -1
- data/lib/spree/testing_support/factories/shipment_factory.rb +1 -1
- data/lib/spree/testing_support/factories/stock_item_factory.rb +5 -1
- data/lib/spree/testing_support/factories/stock_package_factory.rb +1 -1
- data/lib/spree/testing_support/factories/user_factory.rb +1 -1
- data/lib/spree/testing_support/factories/zone_factory.rb +1 -1
- data/lib/spree/testing_support/order_walkthrough.rb +3 -3
- data/lib/spree/testing_support/shared_examples/calculator.rb +10 -0
- data/lib/spree/testing_support/shared_examples/order_factory.rb +141 -0
- data/lib/spree/testing_support/shared_examples/working_factory.rb +15 -0
- data/lib/tasks/colorado_delivery_fee.rake +2 -2
- data/solidus_core.gemspec +7 -10
- metadata +107 -109
- data/app/jobs/spree/promotion_code_batch_job.rb +0 -26
- data/app/mailers/spree/promotion_code_batch_mailer.rb +0 -15
- data/app/models/spree/calculator/distributed_amount.rb +0 -33
- data/app/models/spree/calculator/flat_percent_item_total.rb +0 -23
- data/app/models/spree/calculator/flexi_rate.rb +0 -22
- data/app/models/spree/calculator/percent_on_line_item.rb +0 -13
- data/app/models/spree/calculator/tiered_flat_rate.rb +0 -52
- data/app/models/spree/calculator/tiered_percent.rb +0 -62
- data/app/models/spree/line_item_action.rb +0 -8
- data/app/models/spree/order_promotion.rb +0 -27
- data/app/models/spree/promotion/actions/create_adjustment.rb +0 -81
- data/app/models/spree/promotion/actions/create_item_adjustments.rb +0 -98
- data/app/models/spree/promotion/actions/create_quantity_adjustments.rb +0 -139
- data/app/models/spree/promotion/actions/free_shipping.rb +0 -59
- data/app/models/spree/promotion/order_adjustments_recalculator.rb +0 -92
- data/app/models/spree/promotion/rules/first_order.rb +0 -38
- data/app/models/spree/promotion/rules/first_repeat_purchase_since.rb +0 -36
- data/app/models/spree/promotion/rules/item_total.rb +0 -86
- data/app/models/spree/promotion/rules/nth_order.rb +0 -45
- data/app/models/spree/promotion/rules/one_use_per_user.rb +0 -25
- data/app/models/spree/promotion/rules/option_value.rb +0 -50
- data/app/models/spree/promotion/rules/product.rb +0 -86
- data/app/models/spree/promotion/rules/store.rb +0 -26
- data/app/models/spree/promotion/rules/taxon.rb +0 -91
- data/app/models/spree/promotion/rules/user.rb +0 -34
- data/app/models/spree/promotion/rules/user_logged_in.rb +0 -20
- data/app/models/spree/promotion/rules/user_role.rb +0 -45
- data/app/models/spree/promotion.rb +0 -271
- data/app/models/spree/promotion_action.rb +0 -47
- data/app/models/spree/promotion_category.rb +0 -8
- data/app/models/spree/promotion_chooser.rb +0 -34
- data/app/models/spree/promotion_code/batch_builder.rb +0 -64
- data/app/models/spree/promotion_code.rb +0 -54
- data/app/models/spree/promotion_code_batch.rb +0 -27
- data/app/models/spree/promotion_handler/cart.rb +0 -75
- data/app/models/spree/promotion_handler/coupon.rb +0 -123
- data/app/models/spree/promotion_handler/page.rb +0 -26
- data/app/models/spree/promotion_handler/shipping.rb +0 -61
- data/app/models/spree/promotion_rule.rb +0 -55
- data/app/models/spree/promotion_rule_role.rb +0 -8
- data/app/models/spree/promotion_rule_store.rb +0 -10
- data/app/models/spree/promotion_rule_taxon.rb +0 -8
- data/app/models/spree/promotion_rule_user.rb +0 -10
- data/app/views/spree/promotion_code_batch_mailer/promotion_code_batch_errored.text.erb +0 -2
- data/app/views/spree/promotion_code_batch_mailer/promotion_code_batch_finished.text.erb +0 -2
- data/bin/rails +0 -13
- data/db/migrate/20161017102621_create_spree_promotion_code_batch.rb +0 -38
- data/db/migrate/20180202190713_create_promotion_rule_stores.rb +0 -12
- data/db/migrate/20180328172631_add_join_characters_to_promotion_code_batch.rb +0 -11
- data/db/migrate/20190106184413_remove_code_from_spree_promotions.rb +0 -42
- data/db/migrate/20220317165036_set_promotions_with_any_policy_to_all_if_possible.rb +0 -20
- data/db/migrate/20230322085416_remove_match_policy_from_spree_promotion.rb +0 -5
- data/db/migrate/20230325132905_remove_unused_columns_from_promotion_rules.rb +0 -6
- data/db/migrate/20230325161633_drop_unused_promo_action_line_items.rb +0 -13
- data/lib/generators/spree/dummy/templates/rails/script/rails +0 -6
- data/lib/solidus/migrations/promotions_with_code_handlers.rb +0 -66
- data/lib/spree/permission_sets/promotion_display.rb +0 -25
- data/lib/spree/permission_sets/promotion_management.rb +0 -25
- data/lib/spree/testing_support/factories/order_promotion_factory.rb +0 -8
- data/lib/spree/testing_support/factories/promotion_category_factory.rb +0 -7
- data/lib/spree/testing_support/factories/promotion_code_factory.rb +0 -8
- data/lib/spree/testing_support/factories/promotion_factory.rb +0 -98
@@ -0,0 +1,61 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Spree
|
4
|
+
module Core
|
5
|
+
class StateMachines
|
6
|
+
# Payments' state machine
|
7
|
+
#
|
8
|
+
# for each event the following instance methods are dynamically implemented:
|
9
|
+
# #<event_name>
|
10
|
+
# #<event_name>!
|
11
|
+
# #can_<event_name>?
|
12
|
+
#
|
13
|
+
# for each state the following instance methods are implemented:
|
14
|
+
# #<state_name>?
|
15
|
+
#
|
16
|
+
module Payment
|
17
|
+
extend ActiveSupport::Concern
|
18
|
+
|
19
|
+
included do
|
20
|
+
state_machine initial: :checkout do
|
21
|
+
# With card payments, happens before purchase or authorization happens
|
22
|
+
#
|
23
|
+
# Setting it after creating a profile and authorizing a full amount will
|
24
|
+
# prevent the payment from being authorized again once Order transitions
|
25
|
+
# to complete
|
26
|
+
event :started_processing do
|
27
|
+
transition from: [:checkout, :pending, :completed, :processing], to: :processing
|
28
|
+
end
|
29
|
+
# When processing during checkout fails
|
30
|
+
event :failure do
|
31
|
+
transition from: [:pending, :processing], to: :failed
|
32
|
+
end
|
33
|
+
# With card payments this represents authorizing the payment
|
34
|
+
event :pend do
|
35
|
+
transition from: [:checkout, :processing], to: :pending
|
36
|
+
end
|
37
|
+
# With card payments this represents completing a purchase or capture transaction
|
38
|
+
event :complete do
|
39
|
+
transition from: [:processing, :pending, :checkout], to: :completed
|
40
|
+
end
|
41
|
+
event :void do
|
42
|
+
transition from: [:pending, :processing, :completed, :checkout], to: :void
|
43
|
+
end
|
44
|
+
# when the card brand isnt supported
|
45
|
+
event :invalidate do
|
46
|
+
transition from: [:checkout], to: :invalid
|
47
|
+
end
|
48
|
+
|
49
|
+
after_transition do |payment, transition|
|
50
|
+
payment.state_changes.create!(
|
51
|
+
previous_state: transition.from,
|
52
|
+
next_state: transition.to,
|
53
|
+
name: 'payment'
|
54
|
+
)
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
@@ -0,0 +1,33 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Spree
|
4
|
+
module Core
|
5
|
+
class StateMachines
|
6
|
+
# Reimbursement' state machine
|
7
|
+
#
|
8
|
+
# for each event the following instance methods are dynamically implemented:
|
9
|
+
# #<event_name>
|
10
|
+
# #<event_name>!
|
11
|
+
# #can_<event_name>?
|
12
|
+
#
|
13
|
+
# for each state the following instance methods are implemented:
|
14
|
+
# #<state_name>?
|
15
|
+
#
|
16
|
+
module Reimbursement
|
17
|
+
extend ActiveSupport::Concern
|
18
|
+
|
19
|
+
included do
|
20
|
+
state_machine :reimbursement_status, initial: :pending do
|
21
|
+
event :errored do
|
22
|
+
transition to: :errored, from: [:pending, :errored]
|
23
|
+
end
|
24
|
+
|
25
|
+
event :reimbursed do
|
26
|
+
transition to: :reimbursed, from: [:pending, :errored]
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
@@ -0,0 +1,32 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Spree
|
4
|
+
module Core
|
5
|
+
class StateMachines
|
6
|
+
# Return Authorizations' state machine
|
7
|
+
#
|
8
|
+
# for each event the following instance methods are dynamically implemented:
|
9
|
+
# #<event_name>
|
10
|
+
# #<event_name>!
|
11
|
+
# #can_<event_name>?
|
12
|
+
#
|
13
|
+
# for each state the following instance methods are implemented:
|
14
|
+
# #<state_name>?
|
15
|
+
#
|
16
|
+
module ReturnAuthorization
|
17
|
+
extend ActiveSupport::Concern
|
18
|
+
|
19
|
+
included do
|
20
|
+
state_machine initial: :authorized do
|
21
|
+
before_transition to: :canceled, do: :cancel_return_items
|
22
|
+
|
23
|
+
event :cancel do
|
24
|
+
transition to: :canceled, from: :authorized,
|
25
|
+
if: lambda { |return_authorization| return_authorization.can_cancel_return_items? }
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
@@ -0,0 +1,51 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Spree
|
4
|
+
module Core
|
5
|
+
class StateMachines
|
6
|
+
module ReturnItem
|
7
|
+
# Return Items' acceptance status state machine
|
8
|
+
#
|
9
|
+
# for each event the following instance methods are dynamically implemented:
|
10
|
+
# #<event_name>
|
11
|
+
# #<event_name>!
|
12
|
+
# #can_<event_name>?
|
13
|
+
#
|
14
|
+
# for each state the following instance methods are implemented:
|
15
|
+
# #<state_name>?
|
16
|
+
#
|
17
|
+
module AcceptanceStatus
|
18
|
+
extend ActiveSupport::Concern
|
19
|
+
|
20
|
+
included do
|
21
|
+
state_machine :acceptance_status, initial: :pending do
|
22
|
+
event :attempt_accept do
|
23
|
+
transition to: :accepted, from: :accepted
|
24
|
+
transition to: :accepted, from: :pending, if: ->(return_item) { return_item.eligible_for_return? }
|
25
|
+
transition to: :manual_intervention_required, from: :pending, if: ->(return_item) { return_item.requires_manual_intervention? }
|
26
|
+
transition to: :rejected, from: :pending
|
27
|
+
end
|
28
|
+
|
29
|
+
# bypasses eligibility checks
|
30
|
+
event :accept do
|
31
|
+
transition to: :accepted, from: [:accepted, :pending, :manual_intervention_required]
|
32
|
+
end
|
33
|
+
|
34
|
+
# bypasses eligibility checks
|
35
|
+
event :reject do
|
36
|
+
transition to: :rejected, from: [:accepted, :pending, :manual_intervention_required]
|
37
|
+
end
|
38
|
+
|
39
|
+
# bypasses eligibility checks
|
40
|
+
event :require_manual_intervention do
|
41
|
+
transition to: :manual_intervention_required, from: [:accepted, :pending, :manual_intervention_required]
|
42
|
+
end
|
43
|
+
|
44
|
+
after_transition any => any, do: :persist_acceptance_status_errors
|
45
|
+
end
|
46
|
+
end
|
47
|
+
end
|
48
|
+
end
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|
@@ -0,0 +1,42 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Spree
|
4
|
+
module Core
|
5
|
+
class StateMachines
|
6
|
+
module ReturnItem
|
7
|
+
# Return Items' reception status state machine
|
8
|
+
#
|
9
|
+
# for each event the following instance methods are dynamically implemented:
|
10
|
+
# #<event_name>
|
11
|
+
# #<event_name>!
|
12
|
+
# #can_<event_name>?
|
13
|
+
#
|
14
|
+
# for each state the following instance methods are implemented:
|
15
|
+
# #<state_name>?
|
16
|
+
#
|
17
|
+
module ReceptionStatus
|
18
|
+
extend ActiveSupport::Concern
|
19
|
+
|
20
|
+
included do
|
21
|
+
state_machine :reception_status, initial: :awaiting do
|
22
|
+
after_transition to: ::Spree::ReturnItem::COMPLETED_RECEPTION_STATUSES, do: :attempt_accept, if: :can_attempt_accept?
|
23
|
+
after_transition to: ::Spree::ReturnItem::COMPLETED_RECEPTION_STATUSES, do: :check_unexchange
|
24
|
+
after_transition to: :received, do: :process_inventory_unit!
|
25
|
+
|
26
|
+
event(:cancel) { transition to: :cancelled, from: :awaiting }
|
27
|
+
|
28
|
+
event(:receive) { transition to: :received, from: ::Spree::ReturnItem::INTERMEDIATE_RECEPTION_STATUSES + [:awaiting] }
|
29
|
+
event(:unexchange) { transition to: :unexchanged, from: [:awaiting] }
|
30
|
+
event(:give) { transition to: :given_to_customer, from: :awaiting }
|
31
|
+
event(:lost) { transition to: :lost_in_transit, from: :awaiting }
|
32
|
+
event(:wrong_item_shipped) { transition to: :shipped_wrong_item, from: :awaiting }
|
33
|
+
event(:short_shipped) { transition to: :short_shipped, from: :awaiting }
|
34
|
+
event(:in_transit) { transition to: :in_transit, from: :awaiting }
|
35
|
+
event(:expired) { transition to: :expired, from: :awaiting }
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
40
|
+
end
|
41
|
+
end
|
42
|
+
end
|
@@ -0,0 +1,58 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Spree
|
4
|
+
module Core
|
5
|
+
class StateMachines
|
6
|
+
# Shipments' state machine
|
7
|
+
#
|
8
|
+
# for each event the following instance methods are dynamically implemented:
|
9
|
+
# #<event_name>
|
10
|
+
# #<event_name>!
|
11
|
+
# #can_<event_name>?
|
12
|
+
#
|
13
|
+
# for each state the following instance methods are implemented:
|
14
|
+
# #<state_name>?
|
15
|
+
#
|
16
|
+
module Shipment
|
17
|
+
extend ActiveSupport::Concern
|
18
|
+
|
19
|
+
included do
|
20
|
+
state_machine initial: :pending, use_transactions: false do
|
21
|
+
event :ready do
|
22
|
+
transition from: :pending, to: :shipped, if: :can_transition_from_pending_to_shipped?
|
23
|
+
transition from: :pending, to: :ready, if: :can_transition_from_pending_to_ready?
|
24
|
+
end
|
25
|
+
|
26
|
+
event :pend do
|
27
|
+
transition from: :ready, to: :pending
|
28
|
+
end
|
29
|
+
|
30
|
+
event :ship do
|
31
|
+
transition from: [:ready, :canceled], to: :shipped
|
32
|
+
end
|
33
|
+
after_transition to: :shipped, do: :after_ship
|
34
|
+
|
35
|
+
event :cancel do
|
36
|
+
transition to: :canceled, from: [:pending, :ready]
|
37
|
+
end
|
38
|
+
after_transition to: :canceled, do: :after_cancel
|
39
|
+
|
40
|
+
event :resume do
|
41
|
+
transition from: :canceled, to: :ready, if: :can_transition_from_canceled_to_ready?
|
42
|
+
transition from: :canceled, to: :pending
|
43
|
+
end
|
44
|
+
after_transition from: :canceled, to: [:pending, :ready, :shipped], do: :after_resume
|
45
|
+
|
46
|
+
after_transition do |shipment, transition|
|
47
|
+
shipment.state_changes.create!(
|
48
|
+
previous_state: transition.from,
|
49
|
+
next_state: transition.to,
|
50
|
+
name: 'shipment'
|
51
|
+
)
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
@@ -21,9 +21,6 @@ module Spree
|
|
21
21
|
|
22
22
|
scope :with_payment_profile, -> { where('gateway_customer_profile_id IS NOT NULL') }
|
23
23
|
|
24
|
-
# needed for some of the ActiveMerchant gateways (eg. SagePay)
|
25
|
-
alias_attribute :brand, :cc_type
|
26
|
-
|
27
24
|
# Taken from ActiveMerchant
|
28
25
|
# https://github.com/activemerchant/active_merchant/blob/2f2acd4696e8de76057b5ed670b9aa022abc1187/lib/active_merchant/billing/credit_card_methods.rb#L5
|
29
26
|
CARD_TYPES = {
|
@@ -95,6 +92,12 @@ module Spree
|
|
95
92
|
end
|
96
93
|
end
|
97
94
|
|
95
|
+
# needed for some of the ActiveMerchant gateways (eg. SagePay)
|
96
|
+
alias_attribute :brand, :cc_type
|
97
|
+
|
98
|
+
# Rails 7.1+ won't use the custom setter with alias_attribute
|
99
|
+
alias_method :brand=, :cc_type=
|
100
|
+
|
98
101
|
# Sets the last digits field based on the assigned credit card number.
|
99
102
|
def set_last_digits
|
100
103
|
self.last_digits ||= number.to_s.length <= 4 ? number : number.to_s.slice(-4..)
|
@@ -152,12 +155,12 @@ module Spree
|
|
152
155
|
# card that represents this credit card
|
153
156
|
def to_active_merchant
|
154
157
|
ActiveMerchant::Billing::CreditCard.new(
|
155
|
-
number
|
156
|
-
month
|
157
|
-
year
|
158
|
-
verification_value
|
159
|
-
first_name
|
160
|
-
last_name:
|
158
|
+
number:,
|
159
|
+
month:,
|
160
|
+
year:,
|
161
|
+
verification_value:,
|
162
|
+
first_name:,
|
163
|
+
last_name:
|
161
164
|
)
|
162
165
|
end
|
163
166
|
|
@@ -0,0 +1,40 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Spree
|
4
|
+
class DeprecatedConfigurableClass
|
5
|
+
def self.new(*_args, &_block)
|
6
|
+
@deprecation_proxy ||= DeprecationProxy.new
|
7
|
+
end
|
8
|
+
|
9
|
+
def self.method_missing(method_name, *args, &block)
|
10
|
+
@deprecation_proxy ||= DeprecationProxy.new
|
11
|
+
@deprecation_proxy.send(method_name, args, block)
|
12
|
+
end
|
13
|
+
|
14
|
+
def self.respond_to_missing?(_method_name, _include_private = false)
|
15
|
+
true
|
16
|
+
end
|
17
|
+
|
18
|
+
class DeprecationProxy
|
19
|
+
def method_missing(_method_name, *_args, &_block)
|
20
|
+
issue_deprecation_warning
|
21
|
+
self
|
22
|
+
end
|
23
|
+
|
24
|
+
def respond_to_missing?(_method_name, _include_private = false)
|
25
|
+
true
|
26
|
+
end
|
27
|
+
|
28
|
+
private
|
29
|
+
|
30
|
+
def issue_deprecation_warning
|
31
|
+
Spree.deprecator.warn(
|
32
|
+
<<-WARNING
|
33
|
+
It appears you are using Solidus' Legacy promotion system. This system has been extracted into the
|
34
|
+
`solidus_legacy_promotions` gem. Please add the gem to your Gemfile and follow in the instructions in the README.
|
35
|
+
WARNING
|
36
|
+
)
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
40
|
+
end
|
@@ -107,14 +107,14 @@ module Spree
|
|
107
107
|
# We order by state, because `'backordered' < 'on_hand'`.
|
108
108
|
current_shipment.
|
109
109
|
inventory_units.
|
110
|
-
where(variant:
|
110
|
+
where(variant:).
|
111
111
|
order(state: :asc).
|
112
112
|
limit(new_on_hand_quantity).
|
113
113
|
update_all(shipment_id: desired_shipment.id, state: :on_hand)
|
114
114
|
|
115
115
|
current_shipment.
|
116
116
|
inventory_units.
|
117
|
-
where(variant:
|
117
|
+
where(variant:).
|
118
118
|
order(state: :asc).
|
119
119
|
limit(quantity - new_on_hand_quantity).
|
120
120
|
update_all(shipment_id: desired_shipment.id, state: :backordered)
|
@@ -127,7 +127,7 @@ module Spree
|
|
127
127
|
ActiveRecord::Base.transaction do
|
128
128
|
current_shipment.
|
129
129
|
inventory_units.
|
130
|
-
where(variant:
|
130
|
+
where(variant:).
|
131
131
|
order(state: :asc).
|
132
132
|
limit(quantity).
|
133
133
|
update_all(shipment_id: desired_shipment.id)
|
@@ -145,7 +145,7 @@ module Spree
|
|
145
145
|
if current_stock_location != desired_stock_location
|
146
146
|
0
|
147
147
|
else
|
148
|
-
current_shipment.inventory_units.where(variant:
|
148
|
+
current_shipment.inventory_units.where(variant:).on_hand.count
|
149
149
|
end
|
150
150
|
end
|
151
151
|
|
@@ -64,7 +64,7 @@ module Spree
|
|
64
64
|
# stock location that is associated with this inventory unit's variant
|
65
65
|
def find_stock_item
|
66
66
|
Spree::StockItem.where(stock_location_id: shipment.stock_location_id,
|
67
|
-
variant_id:
|
67
|
+
variant_id:).first
|
68
68
|
end
|
69
69
|
|
70
70
|
# @return [Spree::ReturnItem] a valid return item for this inventory unit
|
@@ -114,7 +114,7 @@ module Spree
|
|
114
114
|
|
115
115
|
def ensure_can_destroy
|
116
116
|
if !backordered? && !on_hand?
|
117
|
-
errors.add(:state, :cannot_destroy, state:
|
117
|
+
errors.add(:state, :cannot_destroy, state:)
|
118
118
|
throw :abort
|
119
119
|
end
|
120
120
|
|
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class Spree::ItemTotal
|
4
|
+
def initialize(item)
|
5
|
+
@item = item
|
6
|
+
end
|
7
|
+
|
8
|
+
def recalculate!
|
9
|
+
tax_adjustments = item.adjustments.select do |adjustment|
|
10
|
+
adjustment.tax? && !adjustment.marked_for_destruction?
|
11
|
+
end
|
12
|
+
|
13
|
+
# Included tax adjustments are those which are included in the price.
|
14
|
+
# These ones should not affect the eventual total price.
|
15
|
+
#
|
16
|
+
# Additional tax adjustments are the opposite, affecting the final total.
|
17
|
+
item.included_tax_total = tax_adjustments.select(&:included?).sum(&:amount)
|
18
|
+
item.additional_tax_total = tax_adjustments.reject(&:included?).sum(&:amount)
|
19
|
+
|
20
|
+
item.adjustment_total = item.adjustments.reject { |adjustment|
|
21
|
+
adjustment.marked_for_destruction? || adjustment.included?
|
22
|
+
}.sum(&:amount)
|
23
|
+
end
|
24
|
+
|
25
|
+
private
|
26
|
+
|
27
|
+
attr_reader :item
|
28
|
+
end
|
@@ -10,6 +10,8 @@ module Spree
|
|
10
10
|
# promotion system.
|
11
11
|
#
|
12
12
|
class LineItem < Spree::Base
|
13
|
+
include Metadata
|
14
|
+
|
13
15
|
belongs_to :order, class_name: "Spree::Order", inverse_of: :line_items, touch: true, optional: true
|
14
16
|
belongs_to :variant, -> { with_discarded }, class_name: "Spree::Variant", inverse_of: :line_items, optional: true
|
15
17
|
belongs_to :tax_category, class_name: "Spree::TaxCategory", optional: true
|
@@ -19,9 +21,6 @@ module Spree
|
|
19
21
|
has_many :adjustments, as: :adjustable, inverse_of: :adjustable, dependent: :destroy
|
20
22
|
has_many :inventory_units, inverse_of: :line_item
|
21
23
|
|
22
|
-
has_many :line_item_actions, dependent: :destroy
|
23
|
-
has_many :actions, through: :line_item_actions
|
24
|
-
|
25
24
|
before_validation :normalize_quantity
|
26
25
|
before_validation :set_required_attributes
|
27
26
|
|
@@ -39,6 +38,7 @@ module Spree
|
|
39
38
|
before_destroy :destroy_inventory_units
|
40
39
|
|
41
40
|
delegate :name, :description, :sku, :should_track_inventory?, to: :variant
|
41
|
+
delegate :tax_category, :tax_category_id, to: :variant, prefix: true
|
42
42
|
delegate :currency, to: :order, allow_nil: true
|
43
43
|
|
44
44
|
attr_accessor :target_shipment, :price_currency
|
@@ -62,7 +62,7 @@ module Spree
|
|
62
62
|
# @return [BigDecimal] the amount of this item, taking into consideration
|
63
63
|
# all non-tax adjustments.
|
64
64
|
def total_before_tax
|
65
|
-
amount + adjustments.
|
65
|
+
amount + adjustments.reject(&:tax?).sum(&:amount)
|
66
66
|
end
|
67
67
|
|
68
68
|
# @return [BigDecimal] the amount of this line item before VAT tax
|
@@ -131,6 +131,24 @@ module Spree
|
|
131
131
|
Spree::Config.pricing_options_class.from_line_item(self)
|
132
132
|
end
|
133
133
|
|
134
|
+
# @return [Spree::TaxCategory] the variant's tax category
|
135
|
+
#
|
136
|
+
# This returns the variant's tax category if the tax category ID on the line_item is nil. It looks
|
137
|
+
# like an association, but really is an override.
|
138
|
+
#
|
139
|
+
def tax_category
|
140
|
+
super || variant_tax_category
|
141
|
+
end
|
142
|
+
|
143
|
+
# @return [Integer] the variant's tax category ID
|
144
|
+
#
|
145
|
+
# This returns the variant's tax category ID if the tax category ID on the line_id is nil. It looks
|
146
|
+
# like an association, but really is an override.
|
147
|
+
#
|
148
|
+
def tax_category_id
|
149
|
+
super || variant_tax_category_id
|
150
|
+
end
|
151
|
+
|
134
152
|
private
|
135
153
|
|
136
154
|
# Sets the quantity to zero if it is nil or less than zero.
|
@@ -0,0 +1,120 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Spree
|
4
|
+
# Spree::Money is a relatively thin wrapper around Monetize which handles
|
5
|
+
# formatting via Spree::Config.
|
6
|
+
class Money
|
7
|
+
include Comparable
|
8
|
+
DifferentCurrencyError = Class.new(StandardError)
|
9
|
+
|
10
|
+
class << self
|
11
|
+
attr_accessor :default_formatting_rules
|
12
|
+
|
13
|
+
def parse(amount, currency = Spree::Config[:currency])
|
14
|
+
new(parse_to_money(amount, currency))
|
15
|
+
end
|
16
|
+
|
17
|
+
# @api private
|
18
|
+
def parse_to_money(amount, currency)
|
19
|
+
::Monetize.parse(amount, currency)
|
20
|
+
end
|
21
|
+
end
|
22
|
+
self.default_formatting_rules = {
|
23
|
+
# Ruby money currently has this as false, which is wrong for the vast
|
24
|
+
# majority of locales.
|
25
|
+
sign_before_symbol: true
|
26
|
+
}
|
27
|
+
|
28
|
+
attr_reader :money
|
29
|
+
|
30
|
+
delegate :cents, :currency, :to_d, :zero?, to: :money
|
31
|
+
|
32
|
+
# @param amount [Money, #to_s] the value of the money object
|
33
|
+
# @param options [Hash] the default options for formatting the money object See #format
|
34
|
+
def initialize(amount, options = {})
|
35
|
+
if amount.is_a?(::Money)
|
36
|
+
@money = amount
|
37
|
+
else
|
38
|
+
currency = options[:currency] || Spree::Config[:currency]
|
39
|
+
|
40
|
+
@money = Monetize.from_string(amount, currency)
|
41
|
+
end
|
42
|
+
@options = Spree::Money.default_formatting_rules.merge(options)
|
43
|
+
end
|
44
|
+
|
45
|
+
# @return [String] the value of this money object formatted according to
|
46
|
+
# its options
|
47
|
+
def to_s
|
48
|
+
format
|
49
|
+
end
|
50
|
+
|
51
|
+
# @param options [Hash, String] the options for formatting the money object
|
52
|
+
# @option options [Boolean] with_currency when true, show the currency
|
53
|
+
# @option options [Boolean] no_cents when true, round to the closest dollar
|
54
|
+
# @option options [String] decimal_mark the mark for delimiting the
|
55
|
+
# decimals
|
56
|
+
# @option options [String, false, nil] thousands_separator the character to
|
57
|
+
# delimit powers of 1000, if one is desired, otherwise false or nil
|
58
|
+
# @option options [Boolean] sign_before_symbol when true the sign of the
|
59
|
+
# value comes before the currency symbol
|
60
|
+
# @option options [:before, :after] symbol_position the position of the
|
61
|
+
# currency symbol
|
62
|
+
# @return [String] the value of this money object formatted according to
|
63
|
+
# its options
|
64
|
+
def format(options = {})
|
65
|
+
@money.format(@options.merge(options))
|
66
|
+
end
|
67
|
+
|
68
|
+
# @note If you pass in options, ensure you pass in the { html_wrap: true } as well.
|
69
|
+
# @param options [Hash] additional formatting options
|
70
|
+
# @return [String] the value of this money object formatted according to
|
71
|
+
# its options and any additional options, by default with html_wrap.
|
72
|
+
def to_html(options = { html_wrap: true })
|
73
|
+
output = format(options)
|
74
|
+
# Maintain compatibility by checking html option renamed to html_wrap.
|
75
|
+
if options[:html_wrap]
|
76
|
+
output = output.html_safe
|
77
|
+
end
|
78
|
+
output
|
79
|
+
end
|
80
|
+
|
81
|
+
# (see #to_s)
|
82
|
+
def as_json(*)
|
83
|
+
to_s
|
84
|
+
end
|
85
|
+
|
86
|
+
def <=>(other)
|
87
|
+
if !other.respond_to?(:money)
|
88
|
+
raise TypeError, "Can't compare #{other.class} to Spree::Money"
|
89
|
+
end
|
90
|
+
if currency != other.currency
|
91
|
+
# By default, ::Money will try to run a conversion on `other.money` and
|
92
|
+
# try a comparison on that. We do not want any currency conversion to
|
93
|
+
# take place so we'll catch this here and raise an error.
|
94
|
+
raise(
|
95
|
+
DifferentCurrencyError,
|
96
|
+
"Can't compare #{currency} with #{other.currency}"
|
97
|
+
)
|
98
|
+
end
|
99
|
+
@money <=> other.money
|
100
|
+
end
|
101
|
+
|
102
|
+
# Delegates comparison to the internal ruby money instance.
|
103
|
+
#
|
104
|
+
# @see http://www.rubydoc.info/gems/money/Money/Arithmetic#%3D%3D-instance_method
|
105
|
+
def ==(other)
|
106
|
+
raise TypeError, "Can't compare #{other.class} to Spree::Money" if !other.respond_to?(:money)
|
107
|
+
@money == other.money
|
108
|
+
end
|
109
|
+
|
110
|
+
def -(other)
|
111
|
+
raise TypeError, "Can't subtract #{other.class} to Spree::Money" if !other.respond_to?(:money)
|
112
|
+
self.class.new(@money - other.money)
|
113
|
+
end
|
114
|
+
|
115
|
+
def +(other)
|
116
|
+
raise TypeError, "Can't add #{other.class} to Spree::Money" if !other.respond_to?(:money)
|
117
|
+
self.class.new(@money + other.money)
|
118
|
+
end
|
119
|
+
end
|
120
|
+
end
|