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.
Files changed (289) hide show
  1. checksums.yaml +4 -4
  2. data/app/assets/javascripts/spree.js.coffee.erb +12 -3
  3. data/app/helpers/spree/base_helper.rb +4 -1
  4. data/app/helpers/spree/products_helper.rb +37 -6
  5. data/app/mailers/spree/base_mailer.rb +11 -2
  6. data/app/models/concerns/spree/adjustment_source.rb +3 -10
  7. data/app/models/concerns/spree/default_price.rb +7 -1
  8. data/app/models/concerns/spree/named_type.rb +1 -1
  9. data/app/models/concerns/spree/user_api_authentication.rb +7 -1
  10. data/app/models/concerns/spree/user_methods.rb +47 -0
  11. data/app/models/concerns/spree/user_reporting.rb +2 -2
  12. data/app/models/concerns/spree/vat_price_calculation.rb +47 -0
  13. data/app/models/spree/address.rb +8 -7
  14. data/app/models/spree/adjustable/adjuster/base.rb +25 -0
  15. data/app/models/spree/adjustable/adjuster/promotion.rb +41 -0
  16. data/app/models/spree/adjustable/adjuster/tax.rb +26 -0
  17. data/app/models/spree/adjustable/adjustments_updater.rb +22 -45
  18. data/app/models/spree/adjustment.rb +8 -10
  19. data/app/models/spree/app_configuration.rb +5 -2
  20. data/app/models/spree/base.rb +4 -0
  21. data/app/models/spree/calculator.rb +0 -5
  22. data/app/models/spree/calculator/default_tax.rb +2 -10
  23. data/app/models/spree/classification.rb +7 -3
  24. data/app/models/spree/country.rb +14 -3
  25. data/app/models/spree/credit_card.rb +21 -31
  26. data/app/models/spree/customer_return.rb +7 -15
  27. data/app/models/spree/gateway.rb +7 -6
  28. data/app/models/spree/image.rb +1 -1
  29. data/app/models/spree/inventory_unit.rb +9 -6
  30. data/app/models/spree/legacy_user.rb +1 -6
  31. data/app/models/spree/line_item.rb +69 -68
  32. data/app/models/spree/log_entry.rb +1 -4
  33. data/app/models/spree/option_type.rb +15 -6
  34. data/app/models/spree/option_type_prototype.rb +9 -0
  35. data/app/models/spree/option_value.rb +11 -3
  36. data/app/models/spree/option_value_variant.rb +6 -0
  37. data/app/models/spree/order.rb +113 -64
  38. data/app/models/spree/order/checkout.rb +8 -11
  39. data/app/models/spree/order/currency_updater.rb +1 -1
  40. data/app/models/spree/order/store_credit.rb +96 -0
  41. data/app/models/spree/order_contents.rb +6 -1
  42. data/app/models/spree/order_inventory.rb +4 -8
  43. data/app/models/spree/order_promotion.rb +6 -0
  44. data/app/models/spree/order_updater.rb +2 -7
  45. data/app/models/spree/payment.rb +46 -19
  46. data/app/models/spree/payment/gateway_options.rb +8 -4
  47. data/app/models/spree/payment_method.rb +12 -13
  48. data/app/models/spree/payment_method/store_credit.rb +130 -0
  49. data/app/models/spree/preference.rb +1 -1
  50. data/app/models/spree/price.rb +16 -6
  51. data/app/models/spree/product.rb +52 -49
  52. data/app/models/spree/product/scopes.rb +7 -2
  53. data/app/models/spree/product_option_type.rb +7 -2
  54. data/app/models/spree/product_promotion_rule.rb +9 -0
  55. data/app/models/spree/product_property.rb +8 -10
  56. data/app/models/spree/promotion.rb +19 -19
  57. data/app/models/spree/promotion/rules/product.rb +3 -1
  58. data/app/models/spree/promotion/rules/taxon.rb +2 -1
  59. data/app/models/spree/promotion/rules/user.rb +4 -4
  60. data/app/models/spree/promotion_action.rb +3 -3
  61. data/app/models/spree/promotion_category.rb +1 -1
  62. data/app/models/spree/promotion_rule_taxon.rb +9 -0
  63. data/app/models/spree/promotion_rule_user.rb +9 -0
  64. data/app/models/spree/property.rb +2 -1
  65. data/app/models/spree/property_prototype.rb +9 -0
  66. data/app/models/spree/prototype.rb +8 -3
  67. data/app/models/spree/prototype_taxon.rb +9 -0
  68. data/app/models/spree/refund.rb +10 -7
  69. data/app/models/spree/refund_reason.rb +1 -1
  70. data/app/models/spree/reimbursement.rb +12 -16
  71. data/app/models/spree/reimbursement_type/reimbursement_helpers.rb +23 -6
  72. data/app/models/spree/reimbursement_type/store_credit.rb +28 -0
  73. data/app/models/spree/return_authorization.rb +8 -13
  74. data/app/models/spree/return_authorization_reason.rb +1 -1
  75. data/app/models/spree/return_item.rb +13 -12
  76. data/app/models/spree/return_item/eligibility_validator/time_since_purchase.rb +1 -1
  77. data/app/models/spree/role.rb +3 -2
  78. data/app/models/spree/role_user.rb +6 -0
  79. data/app/models/spree/shipment.rb +18 -23
  80. data/app/models/spree/shipment_handler.rb +2 -2
  81. data/app/models/spree/shipping_category.rb +6 -3
  82. data/app/models/spree/shipping_method.rb +11 -10
  83. data/app/models/spree/shipping_method_zone.rb +6 -0
  84. data/app/models/spree/shipping_rate.rb +16 -29
  85. data/app/models/spree/state.rb +3 -2
  86. data/app/models/spree/state_change.rb +1 -1
  87. data/app/models/spree/stock/content_item.rb +10 -12
  88. data/app/models/spree/stock/coordinator.rb +13 -14
  89. data/app/models/spree/stock/estimator.rb +28 -30
  90. data/app/models/spree/stock/inventory_unit_builder.rb +1 -9
  91. data/app/models/spree/stock/packer.rb +1 -1
  92. data/app/models/spree/stock/quantifier.rb +5 -5
  93. data/app/models/spree/stock/splitter/backordered.rb +2 -2
  94. data/app/models/spree/stock_item.rb +12 -18
  95. data/app/models/spree/stock_location.rb +4 -7
  96. data/app/models/spree/stock_movement.rb +11 -9
  97. data/app/models/spree/stock_transfer.rb +11 -12
  98. data/app/models/spree/store.rb +14 -6
  99. data/app/models/spree/store_credit.rb +252 -0
  100. data/app/models/spree/store_credit_category.rb +22 -0
  101. data/app/models/spree/store_credit_event.rb +38 -0
  102. data/app/models/spree/store_credit_type.rb +6 -0
  103. data/app/models/spree/tax_category.rb +3 -8
  104. data/app/models/spree/tax_rate.rb +56 -122
  105. data/app/models/spree/taxon.rb +11 -5
  106. data/app/models/spree/tracker.rb +12 -1
  107. data/app/models/spree/validations/db_maximum_length_validator.rb +2 -1
  108. data/app/models/spree/variant.rb +82 -50
  109. data/app/models/spree/zone.rb +21 -17
  110. data/app/models/spree/zone_member.rb +6 -0
  111. data/app/validators/db_maximum_length_validator.rb +11 -0
  112. data/app/views/layouts/spree/base_mailer.html.erb +38 -781
  113. data/app/views/spree/order_mailer/_adjustment.html.erb +8 -0
  114. data/app/views/spree/order_mailer/_subtotal.html.erb +8 -0
  115. data/app/views/spree/order_mailer/_total.html.erb +8 -0
  116. data/app/views/spree/order_mailer/cancel_email.html.erb +13 -28
  117. data/app/views/spree/order_mailer/cancel_email.text.erb +1 -1
  118. data/app/views/spree/order_mailer/confirm_email.html.erb +49 -63
  119. data/app/views/spree/order_mailer/confirm_email.text.erb +1 -1
  120. data/app/views/spree/shared/_base_mailer_header.html.erb +5 -7
  121. data/app/views/spree/shared/_base_mailer_stylesheets.html.erb +777 -0
  122. data/app/views/spree/shared/_error_messages.html.erb +1 -1
  123. data/app/views/spree/shared/_mailer_line_item.html.erb +12 -0
  124. data/app/views/spree/shipment_mailer/shipped_email.html.erb +21 -14
  125. data/app/views/spree/shipment_mailer/shipped_email.text.erb +3 -3
  126. data/config/initializers/user_class_extensions.rb +7 -38
  127. data/config/locales/en.yml +113 -13
  128. data/config/routes.rb +7 -0
  129. data/db/default/spree/default_reimbursement_type.rb +1 -1
  130. data/db/default/spree/zones.rb +4 -5
  131. data/db/migrate/20150118210639_create_spree_store_credits.rb +24 -0
  132. data/db/migrate/20150118211500_create_spree_store_credit_categories.rb +8 -0
  133. data/db/migrate/20150118212051_create_spree_store_credit_events.rb +17 -0
  134. data/db/migrate/20150118212101_create_spree_store_credit_types.rb +10 -0
  135. data/db/migrate/20150314013438_add_missing_indexes.rb +25 -0
  136. data/db/migrate/20150317174308_remove_duplicated_indexes_from_multi_columns.rb +18 -0
  137. data/db/migrate/20150324104002_remove_user_index_from_spree_state_changes.rb +14 -0
  138. data/db/migrate/20150522071831_add_position_to_spree_payment_methods.rb +5 -0
  139. data/db/migrate/20150626181949_add_taxable_adjustment_total_to_line_item.rb +19 -0
  140. data/db/migrate/20150627090949_migrate_payment_methods_display.rb +12 -0
  141. data/db/migrate/20150714154102_spree_payment_method_store_credits.rb +12 -0
  142. data/db/migrate/20150726141425_rename_has_and_belongs_to_associations_to_model_names.rb +18 -0
  143. data/db/migrate/20150727191614_spree_store_credit_types.rb +11 -0
  144. data/db/migrate/20150819154308_add_discontinued_to_products_and_variants.rb +68 -0
  145. data/db/migrate/20151220072838_remove_shipping_method_id_from_spree_orders.rb +13 -0
  146. data/db/migrate/20160207191757_add_id_column_to_earlier_habtm_tables.rb +16 -0
  147. data/db/migrate/20160219165458_add_indexes.rb +14 -0
  148. data/lib/generators/spree/dummy/templates/rails/database.yml +31 -24
  149. data/lib/generators/spree/dummy/templates/rails/test.rb +2 -1
  150. data/lib/spree/core.rb +16 -0
  151. data/lib/spree/core/controller_helpers/auth.rb +1 -1
  152. data/lib/spree/core/controller_helpers/common.rb +3 -3
  153. data/lib/spree/core/controller_helpers/order.rb +6 -5
  154. data/lib/spree/core/controller_helpers/search.rb +1 -1
  155. data/lib/spree/core/controller_helpers/store.rb +29 -0
  156. data/lib/spree/core/delegate_belongs_to.rb +2 -2
  157. data/lib/spree/core/engine.rb +30 -25
  158. data/lib/spree/core/environment.rb +1 -1
  159. data/lib/spree/core/importer/order.rb +37 -40
  160. data/lib/spree/core/number_generator.rb +52 -0
  161. data/lib/spree/core/product_filters.rb +1 -1
  162. data/lib/spree/core/search/base.rb +4 -3
  163. data/lib/spree/core/version.rb +1 -1
  164. data/lib/spree/localized_number.rb +3 -1
  165. data/lib/spree/permitted_attributes.rb +5 -2
  166. data/lib/spree/testing_support/common_rake.rb +3 -3
  167. data/lib/spree/testing_support/factories.rb +3 -3
  168. data/lib/spree/testing_support/factories/address_factory.rb +1 -1
  169. data/lib/spree/testing_support/factories/country_factory.rb +2 -2
  170. data/lib/spree/testing_support/factories/order_factory.rb +2 -2
  171. data/lib/spree/testing_support/factories/payment_factory.rb +5 -0
  172. data/lib/spree/testing_support/factories/payment_method_factory.rb +8 -0
  173. data/lib/spree/testing_support/factories/promotion_rule_factory.rb +5 -0
  174. data/lib/spree/testing_support/factories/refund_factory.rb +9 -1
  175. data/lib/spree/testing_support/factories/return_authorization_factory.rb +2 -0
  176. data/lib/spree/testing_support/factories/state_factory.rb +2 -2
  177. data/lib/spree/testing_support/factories/store_credit_category_factory.rb +9 -0
  178. data/lib/spree/testing_support/factories/store_credit_event_factory.rb +8 -0
  179. data/lib/spree/testing_support/factories/store_credit_factory.rb +17 -0
  180. data/lib/spree/testing_support/factories/store_credit_type_factory.rb +11 -0
  181. data/lib/spree/testing_support/factories/user_factory.rb +1 -1
  182. data/lib/spree/testing_support/factories/zone_member_factory.rb +6 -0
  183. data/lib/spree/testing_support/microdata.rb +189 -0
  184. data/lib/tasks/core.rake +68 -0
  185. data/lib/tasks/exchanges.rake +2 -2
  186. data/spec/fixtures/microdata.html +22 -0
  187. data/spec/fixtures/microdata_itemref.html +15 -0
  188. data/spec/fixtures/microdata_no_itemscope.html +20 -0
  189. data/spec/helpers/base_helper_spec.rb +64 -1
  190. data/spec/helpers/products_helper_spec.rb +75 -3
  191. data/spec/lib/i18n_spec.rb +2 -2
  192. data/spec/lib/search/base_spec.rb +2 -2
  193. data/spec/lib/spree/core/controller_helpers/auth_spec.rb +4 -2
  194. data/spec/lib/spree/core/controller_helpers/store_spec.rb +56 -0
  195. data/spec/lib/spree/core/importer/order_spec.rb +226 -123
  196. data/spec/lib/spree/core/number_generator_spec.rb +175 -0
  197. data/spec/lib/spree/core_spec.rb +23 -0
  198. data/spec/lib/spree/localized_number_spec.rb +10 -0
  199. data/spec/mailers/order_mailer_spec.rb +11 -13
  200. data/spec/mailers/shipment_mailer_spec.rb +26 -8
  201. data/spec/mailers/test_mailer_spec.rb +15 -1
  202. data/spec/models/option_type_prototype_spec.rb +9 -0
  203. data/spec/models/spree/ability_spec.rb +6 -13
  204. data/spec/models/spree/address_spec.rb +1 -1
  205. data/spec/models/spree/adjustable/adjuster/base_spec.rb +10 -0
  206. data/spec/models/spree/adjustable/adjuster/promotion_spec.rb +211 -0
  207. data/spec/models/spree/adjustable/adjuster/tax_spec.rb +86 -0
  208. data/spec/models/spree/adjustable/adjustments_updater_spec.rb +2 -262
  209. data/spec/models/spree/adjustment_spec.rb +27 -1
  210. data/spec/models/spree/app_configuration_spec.rb +5 -2
  211. data/spec/models/spree/calculator/default_tax_spec.rb +39 -14
  212. data/spec/models/spree/concerns/user_methods_spec.rb +55 -0
  213. data/spec/models/spree/concerns/vat_price_calculation_spec.rb +66 -0
  214. data/spec/models/spree/country_spec.rb +45 -8
  215. data/spec/models/spree/credit_card_spec.rb +8 -8
  216. data/spec/models/spree/customer_return_spec.rb +4 -26
  217. data/spec/models/spree/gateway_spec.rb +7 -0
  218. data/spec/models/spree/image_spec.rb +3 -0
  219. data/spec/models/spree/inventory_unit_spec.rb +1 -18
  220. data/spec/models/spree/line_item_spec.rb +78 -18
  221. data/spec/models/spree/option_type_spec.rb +2 -2
  222. data/spec/models/spree/option_value_spec.rb +8 -3
  223. data/spec/models/spree/order/checkout_spec.rb +49 -39
  224. data/spec/models/spree/order/currency_updater_spec.rb +3 -3
  225. data/spec/models/spree/order/finalizing_spec.rb +0 -3
  226. data/spec/models/spree/order/payment_spec.rb +1 -1
  227. data/spec/models/spree/order/state_machine_spec.rb +1 -6
  228. data/spec/models/spree/order/store_credit_spec.rb +423 -0
  229. data/spec/models/spree/order/updating_spec.rb +2 -2
  230. data/spec/models/spree/order_contents_spec.rb +42 -1
  231. data/spec/models/spree/order_inventory_spec.rb +27 -17
  232. data/spec/models/spree/order_spec.rb +65 -52
  233. data/spec/models/spree/payment/gateway_options_spec.rb +10 -2
  234. data/spec/models/spree/payment/store_credit_spec.rb +60 -0
  235. data/spec/models/spree/payment_method/store_credit_spec.rb +291 -0
  236. data/spec/models/spree/payment_method_spec.rb +22 -14
  237. data/spec/models/spree/payment_spec.rb +37 -44
  238. data/spec/models/spree/price_spec.rb +86 -0
  239. data/spec/models/spree/product/scopes_spec.rb +35 -0
  240. data/spec/models/spree/product_option_type_spec.rb +6 -2
  241. data/spec/models/spree/product_promotion_rule_spec.rb +9 -0
  242. data/spec/models/spree/product_property_spec.rb +11 -0
  243. data/spec/models/spree/product_spec.rb +82 -15
  244. data/spec/models/spree/promotion/actions/create_item_adjustments_spec.rb +1 -1
  245. data/spec/models/spree/promotion/rules/user_spec.rb +8 -0
  246. data/spec/models/spree/promotion_action_spec.rb +1 -1
  247. data/spec/models/spree/promotion_rule_spec.rb +1 -1
  248. data/spec/models/spree/promotion_rule_taxon_spec.rb +9 -0
  249. data/spec/models/spree/promotion_rule_user_spec.rb +9 -0
  250. data/spec/models/spree/promotion_spec.rb +57 -36
  251. data/spec/models/spree/property_prototype_spec.rb +9 -0
  252. data/spec/models/spree/prototype_taxon_spec.rb +9 -0
  253. data/spec/models/spree/refund_reason_spec.rb +7 -0
  254. data/spec/models/spree/reimbursement_spec.rb +3 -30
  255. data/spec/models/spree/reimbursement_tax_calculator_spec.rb +17 -5
  256. data/spec/models/spree/reimbursement_type/store_credit_spec.rb +101 -0
  257. data/spec/models/spree/return_authorization_reason_spec.rb +7 -0
  258. data/spec/models/spree/return_authorization_spec.rb +2 -22
  259. data/spec/models/spree/return_item_spec.rb +50 -1
  260. data/spec/models/spree/returns_calculator_spec.rb +1 -1
  261. data/spec/models/spree/role_spec.rb +7 -0
  262. data/spec/models/spree/shipment_spec.rb +17 -17
  263. data/spec/models/spree/shipping_calculator_spec.rb +2 -2
  264. data/spec/models/spree/shipping_category_spec.rb +14 -0
  265. data/spec/models/spree/shipping_method_spec.rb +9 -2
  266. data/spec/models/spree/shipping_rate_spec.rb +40 -41
  267. data/spec/models/spree/state_spec.rb +12 -1
  268. data/spec/models/spree/stock/content_item_spec.rb +9 -0
  269. data/spec/models/spree/stock/estimator_spec.rb +56 -8
  270. data/spec/models/spree/stock/quantifier_spec.rb +61 -32
  271. data/spec/models/spree/stock_item_spec.rb +19 -1
  272. data/spec/models/spree/store_credit_event_spec.rb +101 -0
  273. data/spec/models/spree/store_credit_spec.rb +786 -0
  274. data/spec/models/spree/store_spec.rb +39 -11
  275. data/spec/models/spree/tax_category_spec.rb +6 -1
  276. data/spec/models/spree/tax_rate_spec.rb +204 -44
  277. data/spec/models/spree/user_spec.rb +105 -38
  278. data/spec/models/spree/variant_spec.rb +281 -9
  279. data/spec/models/spree/zone_member_spec.rb +38 -0
  280. data/spec/models/spree/zone_spec.rb +32 -8
  281. data/spec/spec_helper.rb +3 -0
  282. data/spec/support/concerns/{adjustment_source_spec.rb → adjustment_source.rb} +0 -0
  283. data/spec/support/concerns/{default_price_spec.rb → default_price.rb} +9 -0
  284. data/spec/validators/db_maximum_length_validator_spec.rb +22 -0
  285. data/spree_core.gemspec +5 -6
  286. metadata +99 -36
  287. data/CHANGELOG.md +0 -4
  288. data/app/models/concerns/spree/number_generator.rb +0 -39
  289. data/spec/models/spree/validations/db_maximum_length_validator_spec.rb +0 -24
@@ -1,8 +1,10 @@
1
1
  module Spree
2
2
  class Gateway < PaymentMethod
3
+ FROM_DOLLAR_TO_CENT_RATE = 100.0
4
+
3
5
  delegate :authorize, :purchase, :capture, :void, :credit, to: :provider
4
6
 
5
- validates :name, :type, presence: true
7
+ validates :type, presence: true
6
8
 
7
9
  preference :server, :string, default: 'test'
8
10
  preference :test_mode, :boolean, default: true
@@ -11,11 +13,6 @@ module Spree
11
13
  CreditCard
12
14
  end
13
15
 
14
- # instantiates the selected gateway and configures with the options stored in the database
15
- def self.current
16
- super
17
- end
18
-
19
16
  def provider
20
17
  gateway_options = options
21
18
  gateway_options.delete :login if gateway_options.has_key?(:login) and gateway_options[:login].nil?
@@ -45,6 +42,10 @@ module Spree
45
42
  'gateway'
46
43
  end
47
44
 
45
+ def exchange_multiplier
46
+ FROM_DOLLAR_TO_CENT_RATE
47
+ end
48
+
48
49
  def supports?(source)
49
50
  return true unless provider_class.respond_to? :supports?
50
51
  return false unless source.brand
@@ -14,7 +14,7 @@ module Spree
14
14
 
15
15
  # save the w,h of the original image (from which others can be calculated)
16
16
  # we need to look at the write-queue for images which have not been saved yet
17
- after_post_process :find_dimensions
17
+ before_save :find_dimensions, if: :attachment_updated_at_changed?
18
18
 
19
19
  #used by admin products autocomplete
20
20
  def mini_url
@@ -1,10 +1,13 @@
1
1
  module Spree
2
2
  class InventoryUnit < Spree::Base
3
- belongs_to :variant, class_name: "Spree::Variant", inverse_of: :inventory_units
4
- belongs_to :order, class_name: "Spree::Order", inverse_of: :inventory_units
5
- belongs_to :shipment, class_name: "Spree::Shipment", touch: true, inverse_of: :inventory_units
6
- belongs_to :return_authorization, class_name: "Spree::ReturnAuthorization", inverse_of: :inventory_units
7
- belongs_to :line_item, class_name: "Spree::LineItem", inverse_of: :inventory_units
3
+
4
+ with_options inverse_of: :inventory_units do
5
+ belongs_to :variant, class_name: "Spree::Variant"
6
+ belongs_to :order, class_name: "Spree::Order"
7
+ belongs_to :shipment, class_name: "Spree::Shipment", touch: true
8
+ belongs_to :return_authorization, class_name: "Spree::ReturnAuthorization"
9
+ belongs_to :line_item, class_name: "Spree::LineItem"
10
+ end
8
11
 
9
12
  has_many :return_items, inverse_of: :inventory_unit
10
13
  has_one :original_return_item, class_name: "Spree::ReturnItem", foreign_key: :exchange_inventory_unit_id
@@ -53,7 +56,7 @@ module Spree
53
56
  inventory_units.map do |iu|
54
57
  iu.update_columns(
55
58
  pending: false,
56
- updated_at: Time.now,
59
+ updated_at: Time.current,
57
60
  )
58
61
  end
59
62
  end
@@ -3,17 +3,12 @@ module Spree
3
3
  class LegacyUser < Spree::Base
4
4
  include UserAddress
5
5
  include UserPaymentSource
6
+ include UserMethods
6
7
 
7
8
  self.table_name = 'spree_users'
8
9
 
9
- has_many :orders, foreign_key: :user_id
10
-
11
10
  before_destroy :check_completed_orders
12
11
 
13
- def has_spree_role?(role)
14
- true
15
- end
16
-
17
12
  attr_accessor :password
18
13
  attr_accessor :password_confirmation
19
14
 
@@ -1,8 +1,11 @@
1
1
  module Spree
2
2
  class LineItem < Spree::Base
3
- before_validation :invalid_quantity_check
4
- belongs_to :order, class_name: "Spree::Order", inverse_of: :line_items, touch: true
5
- belongs_to :variant, class_name: "Spree::Variant", inverse_of: :line_items
3
+ before_validation :ensure_valid_quantity
4
+
5
+ with_options inverse_of: :line_items do
6
+ belongs_to :order, class_name: "Spree::Order", touch: true
7
+ belongs_to :variant, class_name: "Spree::Variant"
8
+ end
6
9
  belongs_to :tax_category, class_name: "Spree::TaxCategory"
7
10
 
8
11
  has_one :product, through: :variant
@@ -14,15 +17,12 @@ module Spree
14
17
  before_validation :copy_tax_category
15
18
 
16
19
  validates :variant, presence: true
17
- validates :quantity, numericality: {
18
- only_integer: true,
19
- greater_than: -1,
20
- message: Spree.t('validation.must_be_int')
21
- }
20
+ validates :quantity, numericality: { only_integer: true, message: Spree.t('validation.must_be_int') }
22
21
  validates :price, numericality: true
23
- validates_with Stock::AvailabilityValidator
24
22
 
23
+ validates_with Stock::AvailabilityValidator
25
24
  validate :ensure_proper_currency
25
+
26
26
  before_destroy :update_inventory
27
27
  before_destroy :destroy_inventory_units
28
28
 
@@ -31,7 +31,8 @@ module Spree
31
31
 
32
32
  after_create :update_tax_charge
33
33
 
34
- delegate :name, :description, :sku, :should_track_inventory?, to: :variant
34
+ delegate :name, :description, :sku, :should_track_inventory?, :product, to: :variant
35
+ delegate :tax_zone, to: :order
35
36
 
36
37
  attr_accessor :target_shipment
37
38
 
@@ -40,49 +41,51 @@ module Spree
40
41
 
41
42
  def copy_price
42
43
  if variant
43
- self.price = variant.price if price.nil?
44
+ update_price if price.nil?
44
45
  self.cost_price = variant.cost_price if cost_price.nil?
45
46
  self.currency = variant.currency if currency.nil?
46
47
  end
47
48
  end
48
49
 
50
+ def update_price
51
+ self.price = variant.price_including_vat_for(tax_zone: tax_zone)
52
+ end
53
+
49
54
  def copy_tax_category
50
55
  if variant
51
56
  self.tax_category = variant.tax_category
52
57
  end
53
58
  end
54
59
 
60
+ extend DisplayMoney
61
+ money_methods :amount, :subtotal, :discounted_amount, :final_amount, :total, :price
62
+
63
+ alias single_money display_price
64
+ alias single_display_amount display_price
65
+
55
66
  def amount
56
67
  price * quantity
57
68
  end
69
+
58
70
  alias subtotal amount
59
71
 
60
- def discounted_amount
61
- amount + promo_total
72
+ def taxable_amount
73
+ amount + taxable_adjustment_total
62
74
  end
63
75
 
64
- def discounted_money
65
- Spree::Money.new(discounted_amount, { currency: currency })
66
- end
76
+ alias discounted_money display_discounted_amount
77
+ alias_method :discounted_amount, :taxable_amount
67
78
 
68
79
  def final_amount
69
80
  amount + adjustment_total
70
81
  end
71
- alias total final_amount
72
82
 
73
- def single_money
74
- Spree::Money.new(price, { currency: currency })
75
- end
76
- alias single_display_amount single_money
77
-
78
- def money
79
- Spree::Money.new(amount, { currency: currency })
80
- end
81
- alias display_total money
82
- alias display_amount money
83
+ alias total final_amount
84
+ alias money display_total
83
85
 
84
86
  def invalid_quantity_check
85
- self.quantity = 0 if quantity.nil? || quantity < 0
87
+ warn "`invalid_quantity_check` is deprecated. Use private `ensure_valid_quantity` instead."
88
+ ensure_valid_quantity
86
89
  end
87
90
 
88
91
  def sufficient_stock?
@@ -93,65 +96,63 @@ module Spree
93
96
  !sufficient_stock?
94
97
  end
95
98
 
96
- # Remove product default_scope `deleted_at: nil`
97
- def product
98
- variant.product
99
- end
100
-
101
- # Remove variant default_scope `deleted_at: nil`
102
- def variant
103
- Spree::Variant.unscoped { super }
104
- end
105
-
106
- def options=(options={})
99
+ def options=(options = {})
107
100
  return unless options.present?
108
101
 
109
102
  opts = options.dup # we will be deleting from the hash, so leave the caller's copy intact
110
103
 
111
104
  currency = opts.delete(:currency) || order.try(:currency)
112
105
 
106
+ update_price_from_modifier(currency, opts)
107
+ assign_attributes opts
108
+ end
109
+
110
+ private
111
+
112
+ def ensure_valid_quantity
113
+ self.quantity = 0 if quantity.nil? || quantity < 0
114
+ end
115
+
116
+ def update_price_from_modifier(currency, opts)
113
117
  if currency
114
118
  self.currency = currency
115
- self.price = variant.price_in(currency).amount +
116
- variant.price_modifier_amount_in(currency, opts)
119
+ self.price = variant.price_in(currency).amount +
120
+ variant.price_modifier_amount_in(currency, opts)
117
121
  else
118
- self.price = variant.price +
119
- variant.price_modifier_amount(opts)
122
+ self.price = variant.price +
123
+ variant.price_modifier_amount(opts)
120
124
  end
121
-
122
- self.assign_attributes opts
123
125
  end
124
126
 
125
- private
126
- def update_inventory
127
- if (changed? || target_shipment.present?) && self.order.has_checkout_step?("delivery")
128
- Spree::OrderInventory.new(self.order, self).verify(target_shipment)
129
- end
127
+ def update_inventory
128
+ if (changed? || target_shipment.present?) && order.has_checkout_step?("delivery")
129
+ Spree::OrderInventory.new(order, self).verify(target_shipment)
130
130
  end
131
+ end
131
132
 
132
- def destroy_inventory_units
133
- inventory_units.destroy_all
134
- end
133
+ def destroy_inventory_units
134
+ inventory_units.destroy_all
135
+ end
135
136
 
136
- def update_adjustments
137
- if quantity_changed?
138
- recalculate_adjustments
139
- update_tax_charge # Called to ensure pre_tax_amount is updated.
140
- end
137
+ def update_adjustments
138
+ if quantity_changed?
139
+ recalculate_adjustments
140
+ update_tax_charge # Called to ensure pre_tax_amount is updated.
141
141
  end
142
+ end
142
143
 
143
- def recalculate_adjustments
144
- Adjustable::AdjustmentsUpdater.update(self)
145
- end
144
+ def recalculate_adjustments
145
+ Adjustable::AdjustmentsUpdater.update(self)
146
+ end
146
147
 
147
- def update_tax_charge
148
- Spree::TaxRate.adjust(order, [self])
149
- end
148
+ def update_tax_charge
149
+ Spree::TaxRate.adjust(order, [self])
150
+ end
150
151
 
151
- def ensure_proper_currency
152
- unless currency == order.currency
153
- errors.add(:currency, :must_match_order_currency)
154
- end
152
+ def ensure_proper_currency
153
+ unless currency == order.currency
154
+ errors.add(:currency, :must_match_order_currency)
155
155
  end
156
+ end
156
157
  end
157
158
  end
@@ -7,10 +7,7 @@ module Spree
7
7
  after_rollback :save_anyway
8
8
 
9
9
  def save_anyway
10
- log = Spree::LogEntry.new
11
- log.source = source
12
- log.details = details
13
- log.save!
10
+ Spree::LogEntry.create!(source: source, details: details)
14
11
  end
15
12
 
16
13
  def parsed_details
@@ -2,20 +2,29 @@ module Spree
2
2
  class OptionType < Spree::Base
3
3
  acts_as_list
4
4
 
5
- has_many :option_values, -> { order(:position) }, dependent: :destroy, inverse_of: :option_type
6
- has_many :product_option_types, dependent: :destroy, inverse_of: :option_type
5
+ with_options dependent: :destroy, inverse_of: :option_type do
6
+ has_many :option_values, -> { order(:position) }
7
+ has_many :product_option_types
8
+ end
9
+
7
10
  has_many :products, through: :product_option_types
8
- has_and_belongs_to_many :prototypes, join_table: 'spree_option_types_prototypes'
9
11
 
10
- validates :name, presence: true, uniqueness: true
11
- validates :presentation, presence: true
12
+ has_many :option_type_prototypes, class_name: 'Spree::OptionTypePrototype'
13
+ has_many :prototypes, through: :option_type_prototypes, class_name: 'Spree::Prototype'
12
14
 
13
- default_scope { order("#{self.table_name}.position") }
15
+ with_options presence: true do
16
+ validates :name, uniqueness: { allow_blank: true }
17
+ validates :presentation
18
+ end
19
+
20
+ default_scope { order(:position) }
14
21
 
15
22
  accepts_nested_attributes_for :option_values, reject_if: lambda { |ov| ov[:name].blank? || ov[:presentation].blank? }, allow_destroy: true
16
23
 
17
24
  after_touch :touch_all_products
18
25
 
26
+ private
27
+
19
28
  def touch_all_products
20
29
  products.update_all(updated_at: Time.current)
21
30
  end
@@ -0,0 +1,9 @@
1
+ module Spree
2
+ class OptionTypePrototype < Spree::Base
3
+ belongs_to :option_type, class_name: 'Spree::OptionType'
4
+ belongs_to :prototype, class_name: 'Spree::Prototype'
5
+
6
+ validates :prototype, :option_type, presence: true
7
+ validates :prototype_id, uniqueness: { scope: :option_type_id }, allow_nil: true
8
+ end
9
+ end
@@ -2,15 +2,23 @@ module Spree
2
2
  class OptionValue < Spree::Base
3
3
  belongs_to :option_type, class_name: 'Spree::OptionType', touch: true, inverse_of: :option_values
4
4
  acts_as_list scope: :option_type
5
- has_and_belongs_to_many :variants, join_table: 'spree_option_values_variants', class_name: "Spree::Variant"
6
5
 
7
- validates :name, presence: true, uniqueness: { scope: :option_type_id }
8
- validates :presentation, presence: true
6
+ has_many :option_value_variants, class_name: 'Spree::OptionValueVariant'
7
+ has_many :variants, through: :option_value_variants, class_name: 'Spree::Variant'
8
+
9
+ with_options presence: true do
10
+ validates :name, uniqueness: { scope: :option_type_id, allow_blank: true }
11
+ validates :presentation
12
+ end
9
13
 
10
14
  after_touch :touch_all_variants
11
15
 
16
+ delegate :name, :presentation, to: :option_type, prefix: true, allow_nil: true
17
+
12
18
  self.whitelisted_ransackable_attributes = ['presentation']
13
19
 
20
+ private
21
+
14
22
  def touch_all_variants
15
23
  variants.update_all(updated_at: Time.current)
16
24
  end
@@ -0,0 +1,6 @@
1
+ module Spree
2
+ class OptionValueVariant < Spree::Base
3
+ belongs_to :option_value, class_name: 'Spree::OptionValue'
4
+ belongs_to :variant, class_name: 'Spree::Variant'
5
+ end
6
+ end
@@ -3,7 +3,7 @@ require 'spree/order/checkout'
3
3
 
4
4
  module Spree
5
5
  class Order < Spree::Base
6
- PAYMENT_STATES = %w(balance_due checkout completed credit_owed failed paid pending processing void).freeze
6
+ PAYMENT_STATES = %w(balance_due credit_owed failed paid void).freeze
7
7
  SHIPMENT_STATES = %w(backorder canceled partial pending ready shipped).freeze
8
8
 
9
9
  extend FriendlyId
@@ -12,14 +12,10 @@ module Spree
12
12
  include Spree::Order::Checkout
13
13
  include Spree::Order::CurrencyUpdater
14
14
  include Spree::Order::Payments
15
- include Spree::NumberGenerator
15
+ include Spree::Order::StoreCredit
16
+ include Spree::Core::NumberGenerator.new(prefix: 'R')
16
17
  include Spree::Core::TokenGenerator
17
18
 
18
- def generate_number(options = {})
19
- options[:prefix] ||= 'R'
20
- super(options)
21
- end
22
-
23
19
  extend Spree::DisplayMoney
24
20
  money_methods :outstanding_balance, :item_total, :adjustment_total,
25
21
  :included_tax_total, :additional_tax_total, :tax_total,
@@ -28,6 +24,25 @@ module Spree
28
24
  alias :display_ship_total :display_shipment_total
29
25
  alias_attribute :ship_total, :shipment_total
30
26
 
27
+ MONEY_THRESHOLD = 100_000_000
28
+ MONEY_VALIDATION = {
29
+ presence: true,
30
+ numericality: {
31
+ greater_than: -MONEY_THRESHOLD,
32
+ less_than: MONEY_THRESHOLD,
33
+ allow_blank: true
34
+ },
35
+ format: { with: /\A-?\d+(?:\.\d{1,2})?\z/, allow_blank: true }
36
+ }.freeze
37
+
38
+ POSITIVE_MONEY_VALIDATION = MONEY_VALIDATION.deep_dup.tap do |validation|
39
+ validation.fetch(:numericality)[:greater_than_or_equal_to] = 0
40
+ end.freeze
41
+
42
+ NEGATIVE_MONEY_VALIDATION = MONEY_VALIDATION.deep_dup.tap do |validation|
43
+ validation.fetch(:numericality)[:less_than_or_equal_to] = 0
44
+ end.freeze
45
+
31
46
  checkout_flow do
32
47
  go_to_state :address
33
48
  go_to_state :delivery
@@ -62,12 +77,15 @@ module Spree
62
77
  alias_attribute :shipping_address, :ship_address
63
78
 
64
79
  belongs_to :store, class_name: 'Spree::Store'
65
- has_many :state_changes, as: :stateful, dependent: :destroy
66
- has_many :line_items, -> { order("#{LineItem.table_name}.created_at ASC") }, dependent: :destroy, inverse_of: :order
67
- has_many :payments, dependent: :destroy
68
- has_many :return_authorizations, dependent: :destroy, inverse_of: :order
80
+
81
+ with_options dependent: :destroy do
82
+ has_many :state_changes, as: :stateful
83
+ has_many :line_items, -> { order(:created_at) }, inverse_of: :order
84
+ has_many :payments
85
+ has_many :return_authorizations, inverse_of: :order
86
+ has_many :adjustments, -> { order(:created_at) }, as: :adjustable
87
+ end
69
88
  has_many :reimbursements, inverse_of: :order
70
- has_many :adjustments, -> { order("#{Adjustment.table_name}.created_at ASC") }, as: :adjustable, dependent: :destroy
71
89
  has_many :line_item_adjustments, through: :line_items, source: :adjustments
72
90
  has_many :shipment_adjustments, through: :shipments, source: :adjustments
73
91
  has_many :inventory_units, inverse_of: :order
@@ -80,7 +98,8 @@ module Spree
80
98
  dependent: :destroy,
81
99
  inverse_of: :order
82
100
 
83
- has_and_belongs_to_many :promotions, join_table: 'spree_orders_promotions'
101
+ has_many :order_promotions, class_name: 'Spree::OrderPromotion'
102
+ has_many :promotions, through: :order_promotions, class_name: 'Spree::Promotion'
84
103
 
85
104
  has_many :shipments, dependent: :destroy, inverse_of: :order do
86
105
  def states
@@ -103,12 +122,26 @@ module Spree
103
122
  before_create :link_by_email
104
123
  before_update :homogenize_line_item_currencies, if: :currency_changed?
105
124
 
106
- validates :email, presence: true, if: :require_email
107
- validates :email, email: true, if: :require_email, allow_blank: true
108
- validate :has_available_shipment
109
-
110
- delegate :update_totals, :persist_totals, :to => :updater
125
+ with_options presence: true do
126
+ validates :number, length: { maximum: 32, allow_blank: true }, uniqueness: { allow_blank: true }
127
+ validates :email, length: { maximum: 254, allow_blank: true }, email: { allow_blank: true }, if: :require_email
128
+ validates :state, inclusion: { in: state_machine.states.map { |state| state.name.to_s }, allow_blank: true }
129
+ validates :item_count, numericality: { greater_than_or_equal_to: 0, less_than: 2**31, only_integer: true, allow_blank: true }
130
+ end
131
+ validates :payment_state, inclusion: { in: PAYMENT_STATES, allow_blank: true }
132
+ validates :shipment_state, inclusion: { in: SHIPMENT_STATES, allow_blank: true }
133
+ validates :item_total, POSITIVE_MONEY_VALIDATION
134
+ validates :adjustment_total, MONEY_VALIDATION
135
+ validates :included_tax_total, POSITIVE_MONEY_VALIDATION
136
+ validates :additional_tax_total, POSITIVE_MONEY_VALIDATION
137
+ validates :payment_total, MONEY_VALIDATION
138
+ validates :shipment_total, MONEY_VALIDATION
139
+ validates :promo_total, NEGATIVE_MONEY_VALIDATION
140
+ validates :total, MONEY_VALIDATION
141
+
142
+ delegate :update_totals, :persist_totals, to: :updater
111
143
  delegate :merge!, to: :merger
144
+ delegate :firstname, :lastname, to: :bill_address, prefix: true, allow_nil: true
112
145
 
113
146
  class_attribute :update_hooks
114
147
  self.update_hooks = Set.new
@@ -118,18 +151,12 @@ module Spree
118
151
 
119
152
  scope :created_between, ->(start_date, end_date) { where(created_at: start_date..end_date) }
120
153
  scope :completed_between, ->(start_date, end_date) { where(completed_at: start_date..end_date) }
154
+ scope :complete, -> { where.not(completed_at: nil) }
155
+ scope :incomplete, -> { where(completed_at: nil) }
121
156
 
122
157
  # shows completed orders first, by their completed_at date, then uncompleted orders by their created_at
123
158
  scope :reverse_chronological, -> { order('spree_orders.completed_at IS NULL', completed_at: :desc, created_at: :desc) }
124
159
 
125
- def self.complete
126
- where.not(completed_at: nil)
127
- end
128
-
129
- def self.incomplete
130
- where(completed_at: nil)
131
- end
132
-
133
160
  # Use this method in other gems that wish to register their own custom logic
134
161
  # that should be called after Order#update
135
162
  def self.register_update_hook(hook)
@@ -184,7 +211,7 @@ module Spree
184
211
  # Little hacky fix for #4117
185
212
  # If this wasn't here, order would transition to address state on confirm failure
186
213
  # because there would be no valid payments any more.
187
- state == 'confirm'
214
+ confirm?
188
215
  end
189
216
 
190
217
  def backordered?
@@ -206,10 +233,15 @@ module Spree
206
233
  @updater ||= OrderUpdater.new(self)
207
234
  end
208
235
 
209
- def update!
236
+ def update_with_updater!
210
237
  updater.update
211
238
  end
212
239
 
240
+ def update!
241
+ warn "`update!` is deprecated as it conflicts with update! method of rails. Use `update_with_updater!` instead."
242
+ update_with_updater!
243
+ end
244
+
213
245
  def merger
214
246
  @merger ||= Spree::OrderMerger.new(self)
215
247
  end
@@ -224,7 +256,7 @@ module Spree
224
256
  end
225
257
 
226
258
  def allow_cancel?
227
- return false unless completed? and state != 'canceled'
259
+ return false if !completed? || canceled?
228
260
  shipment_state.nil? || %w{ready backorder pending}.include?(shipment_state)
229
261
  end
230
262
 
@@ -287,8 +319,15 @@ module Spree
287
319
  Spree::TaxRate.adjust(self, shipments) if shipments.any?
288
320
  end
289
321
 
322
+ def update_line_item_prices!
323
+ transaction do
324
+ line_items.each(&:update_price)
325
+ save!
326
+ end
327
+ end
328
+
290
329
  def outstanding_balance
291
- if state == 'canceled'
330
+ if canceled?
292
331
  -1 * payment_total
293
332
  elsif reimbursements.includes(:refunds).size > 0
294
333
  reimbursed = reimbursements.includes(:refunds).inject(0) do |sum, reimbursement|
@@ -308,7 +347,7 @@ module Spree
308
347
 
309
348
  def name
310
349
  if (address = bill_address || ship_address)
311
- "#{address.firstname} #{address.lastname}"
350
+ address.full_name
312
351
  end
313
352
  end
314
353
 
@@ -367,15 +406,7 @@ module Spree
367
406
  end
368
407
 
369
408
  def available_payment_methods
370
- @available_payment_methods ||= (PaymentMethod.available(:front_end) + PaymentMethod.available(:both)).uniq
371
- end
372
-
373
- def billing_firstname
374
- bill_address.try(:firstname)
375
- end
376
-
377
- def billing_lastname
378
- bill_address.try(:lastname)
409
+ @available_payment_methods ||= PaymentMethod.available_on_front_end
379
410
  end
380
411
 
381
412
  def insufficient_stock_lines
@@ -383,11 +414,11 @@ module Spree
383
414
  end
384
415
 
385
416
  ##
386
- # Check to see if any line item variants are soft deleted.
417
+ # Check to see if any line item variants are discontinued.
387
418
  # If so add error and restart checkout.
388
- def ensure_line_item_variants_are_not_deleted
389
- if line_items.any?{ |li| !li.variant || li.variant.paranoia_destroyed? }
390
- errors.add(:base, Spree.t(:deleted_variants_present))
419
+ def ensure_line_item_variants_are_not_discontinued
420
+ if line_items.any?{ |li| !li.variant || li.variant.discontinued? }
421
+ errors.add(:base, Spree.t(:discontinued_variants_present))
391
422
  restart_checkout_flow
392
423
  false
393
424
  else
@@ -406,14 +437,21 @@ module Spree
406
437
  end
407
438
 
408
439
  def empty!
409
- line_items.destroy_all
410
- updater.update_item_count
411
- adjustments.destroy_all
412
- shipments.destroy_all
413
- state_changes.destroy_all
440
+ if completed?
441
+ raise Spree.t(:cannot_empty_completed_order)
442
+ else
443
+ line_items.destroy_all
444
+ updater.update_item_count
445
+ adjustments.destroy_all
446
+ shipments.destroy_all
447
+ state_changes.destroy_all
448
+ order_promotions.destroy_all
414
449
 
415
- update_totals
416
- persist_totals
450
+ update_totals
451
+ persist_totals
452
+ restart_checkout_flow
453
+ self
454
+ end
417
455
  end
418
456
 
419
457
  def has_step?(step)
@@ -478,7 +516,7 @@ module Spree
478
516
  def restart_checkout_flow
479
517
  self.update_columns(
480
518
  state: 'cart',
481
- updated_at: Time.now,
519
+ updated_at: Time.current,
482
520
  )
483
521
  self.next! if self.line_items.size > 0
484
522
  end
@@ -498,7 +536,7 @@ module Spree
498
536
  end
499
537
 
500
538
  def is_risky?
501
- self.payments.risky.count > 0
539
+ payments.risky.size > 0
502
540
  end
503
541
 
504
542
  def canceled_by(user)
@@ -506,7 +544,7 @@ module Spree
506
544
  cancel!
507
545
  self.update_columns(
508
546
  canceler_id: user.id,
509
- canceled_at: Time.now,
547
+ canceled_at: Time.current,
510
548
  )
511
549
  end
512
550
  end
@@ -516,7 +554,7 @@ module Spree
516
554
  approve!
517
555
  self.update_columns(
518
556
  approver_id: user.id,
519
- approved_at: Time.now,
557
+ approved_at: Time.current,
520
558
  )
521
559
  end
522
560
  end
@@ -573,6 +611,20 @@ module Spree
573
611
 
574
612
  private
575
613
 
614
+ def create_store_credit_payment(payment_method, credit, amount)
615
+ payments.create!(
616
+ source: credit,
617
+ payment_method: payment_method,
618
+ amount: amount,
619
+ state: 'checkout',
620
+ response_code: credit.generate_authorization_code
621
+ )
622
+ end
623
+
624
+ def store_credit_amount(credit, total)
625
+ [credit.amount_remaining, total].min
626
+ end
627
+
576
628
  def link_by_email
577
629
  self.email = user.email if self.user
578
630
  end
@@ -588,13 +640,6 @@ module Spree
588
640
  end
589
641
  end
590
642
 
591
- def has_available_shipment
592
- return unless has_step?("delivery")
593
- return unless has_step?(:address) && address?
594
- return unless ship_address && ship_address.valid?
595
- # errors.add(:base, :no_shipping_methods_available) if available_shipping_methods.empty?
596
- end
597
-
598
643
  def ensure_available_shipping_rates
599
644
  if shipments.empty? || shipments.any? { |shipment| shipment.shipping_rates.blank? }
600
645
  # After this point, order redirects back to 'address' state and asks user to pick a proper address
@@ -607,8 +652,12 @@ module Spree
607
652
  def after_cancel
608
653
  shipments.each { |shipment| shipment.cancel! }
609
654
  payments.completed.each { |payment| payment.cancel! }
655
+
656
+ # Free up authorized store credits
657
+ payments.store_credits.pending.each(&:void!)
658
+
610
659
  send_cancel_email
611
- self.update!
660
+ self.update_with_updater!
612
661
  end
613
662
 
614
663
  def send_cancel_email
@@ -621,7 +670,7 @@ module Spree
621
670
  end
622
671
 
623
672
  def use_billing?
624
- @use_billing == true || @use_billing == 'true' || @use_billing == '1'
673
+ use_billing.in?([true, 'true', '1'])
625
674
  end
626
675
 
627
676
  def set_currency