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
@@ -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
@@ -0,0 +1,6 @@
1
+ module Spree
2
+ class StoreCreditType < Spree::Base
3
+ DEFAULT_TYPE_NAME = 'Expiring'.freeze
4
+ has_many :store_credits, class_name: 'Spree::StoreCredit', foreign_key: 'type_id'
5
+ end
6
+ 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
- unless tax_category == self
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
- belongs_to :zone, class_name: "Spree::Zone", inverse_of: :tax_rates
19
- belongs_to :tax_category, class_name: "Spree::TaxCategory", inverse_of: :tax_rates
20
-
21
- validates :amount, presence: true, numericality: true
22
- validates :tax_category_id, presence: true
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
- def self.potential_rates_for_zone(zone)
28
- select("spree_tax_rates.*, spree_zones.default_tax").
29
- joins(:zone).
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
- # Gets the array of TaxRates appropriate for the specified order
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
- when Spree::LineItem then item.discounted_amount
70
- when Spree::Shipment then item.discounted_cost
71
- end
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
- # This method is best described by the documentation on #potentially_applicable?
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
- relevant_items, non_relevant_items = items.partition { |item| tax_categories.include?(item.tax_category) }
86
- Spree::Adjustment.where(adjustable: relevant_items).tax.destroy_all # using destroy_all to ensure adjustment destroy callback fires.
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 { |rate| rate.tax_category == item.tax_category }
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
- # Tax rates can *potentially* be applicable to an order.
103
- # We do not know if they are/aren't until we attempt to apply these rates to
104
- # the items contained within the Order itself.
105
- # For instance, if a rate passes the criteria outlined in this method,
106
- # but then has a tax category that doesn't match against any of the line items
107
- # inside of the order, then that tax rate will not be applicable to anything.
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
- included = included_in_price && default_zone_or_zone_match?(order)
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
- refund = included_in_price && !default_zone_or_zone_match?(item.order)
158
- compute(item) * (refund ? -1 : 1)
90
+ compute(item)
159
91
  end
160
92
 
161
93
  private
162
94
 
163
- def default_zone_or_zone_match?(order)
164
- Zone.default_tax.try(:contains?, order.tax_zone) || order.tax_zone == zone
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 label(adjustment_amount)
168
- label = ""
169
- label << Spree.t(:refund) << ' ' if adjustment_amount < 0
170
- label << (name.present? ? name : tax_category.name) + " "
171
- label << (show_rate_in_label? ? "#{amount * 100}%" : "")
172
- label << " (#{Spree.t(:included_in_price)})" if included_in_price?
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
@@ -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
- has_and_belongs_to_many :prototypes, join_table: :spree_taxons_prototypes
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
- validates :meta_keywords, length: { maximum: 255 }
20
- validates :meta_description, length: { maximum: 255 }
21
- validates :meta_title, length: { maximum: 255 }
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
- self.class.where(id: ancestors.pluck(:id)).update_all(updated_at: Time.now)
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
@@ -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 = where(active: true).first
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
@@ -12,15 +12,22 @@ module Spree
12
12
  :shipping_category_id, :meta_description, :meta_keywords,
13
13
  :shipping_category
14
14
 
15
- has_many :inventory_units, inverse_of: :variant
16
- has_many :line_items, inverse_of: :variant
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 :stock_items, dependent: :destroy, inverse_of: :variant
20
- has_many :stock_locations, through: :stock_items
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 :cost_price, numericality: { greater_than_or_equal_to: 0, allow_nil: true }
36
- validates :price, numericality: { greater_than_or_equal_to: 0, allow_nil: true }
37
- validates_uniqueness_of :sku, allow_blank: true, conditions: -> { where(deleted_at: nil) }
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 self.active(currency = nil)
58
- joins(:prices).where(deleted_at: nil).where('spree_prices.currency' => currency || Spree::Config[:currency]).where('spree_prices.amount IS NOT NULL')
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 } || Spree::Price.new(variant_id: id, 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
- def can_supply?(quantity=1)
206
- Spree::Stock::Quantifier.new(self).can_supply?(quantity)
207
- end
229
+ delegate :total_on_hand, :can_supply?, :backorderable?, to: :quantifier
208
230
 
209
- def total_on_hand
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
- def set_master_out_of_stock
234
- if product.master && product.master.in_stock?
235
- product.master.stock_items.update_all(:backorderable => false)
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
- # Ensures a new variant takes the product master price when price is not supplied
241
- def check_price
242
- if price.nil? && Spree::Config[:require_master_price]
243
- raise 'No master variant found to infer price' unless (product && product.master)
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
- def set_cost_currency
253
- self.cost_currency = Spree::Config[:currency] if cost_currency.nil? || cost_currency.empty?
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
- def create_stock_items
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
- def in_stock_cache_key
263
- "variant-#{id}-in_stock"
264
- end
284
+ def set_cost_currency
285
+ self.cost_currency = Spree::Config[:currency] if cost_currency.blank?
286
+ end
265
287
 
266
- def clear_in_stock_cache
267
- Rails.cache.delete(in_stock_cache_key)
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