spree_core 4.2.5 → 4.3.0.rc1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/app/finders/concerns/spree/product_filterable.rb +9 -0
- data/app/finders/spree/cms_pages/find.rb +41 -0
- data/app/finders/spree/menus/find.rb +11 -0
- data/app/finders/spree/option_values/find_available.rb +20 -0
- data/app/finders/spree/orders/find_complete.rb +14 -2
- data/app/finders/spree/orders/find_current.rb +1 -13
- data/app/finders/spree/product_properties/find_available.rb +20 -0
- data/app/finders/spree/products/find.rb +55 -20
- data/app/finders/spree/stores/find_current.rb +24 -0
- data/app/helpers/spree/base_helper.rb +62 -2
- data/app/helpers/spree/locale_helper.rb +5 -1
- data/app/helpers/spree/products_helper.rb +7 -3
- data/app/models/concerns/spree/display_link.rb +42 -0
- data/app/models/concerns/spree/filter_param.rb +21 -0
- data/app/models/concerns/spree/memoized_data.rb +24 -0
- data/app/models/concerns/spree/multi_store_resource.rb +24 -0
- data/app/models/concerns/spree/product_scopes.rb +59 -8
- data/app/models/concerns/spree/single_store_resource.rb +9 -0
- data/app/models/concerns/spree/user_address.rb +19 -0
- data/app/models/concerns/spree/user_methods.rb +5 -4
- data/app/models/concerns/spree/user_payment_source.rb +1 -1
- data/app/models/spree/ability.rb +3 -1
- data/app/models/spree/address.rb +27 -6
- data/app/models/spree/app_configuration.rb +8 -0
- data/app/models/spree/app_dependencies.rb +18 -6
- data/app/models/spree/base.rb +18 -0
- data/app/models/spree/classification.rb +3 -0
- data/app/models/spree/cms/pages/feature_page.rb +7 -0
- data/app/models/spree/cms/pages/homepage.rb +20 -0
- data/app/models/spree/cms/pages/standard_page.rb +4 -0
- data/app/models/spree/cms/sections/featured_article.rb +29 -0
- data/app/models/spree/cms/sections/hero_image.rb +47 -0
- data/app/models/spree/cms/sections/image_gallery.rb +103 -0
- data/app/models/spree/cms/sections/product_carousel.rb +14 -0
- data/app/models/spree/cms/sections/rich_text_content.rb +13 -0
- data/app/models/spree/cms/sections/side_by_side_images.rb +74 -0
- data/app/models/spree/cms_page.rb +66 -0
- data/app/models/spree/cms_section.rb +57 -0
- data/app/models/spree/country.rb +14 -6
- data/app/models/spree/credit_card.rb +2 -8
- data/app/models/spree/customer_return.rb +8 -2
- data/app/models/spree/icon.rb +9 -0
- data/app/models/spree/image/configuration/active_storage.rb +2 -0
- data/app/models/spree/inventory_unit.rb +1 -1
- data/app/models/spree/line_item.rb +1 -1
- data/app/models/spree/menu.rb +63 -0
- data/app/models/spree/menu_item.rb +76 -0
- data/app/models/spree/option_type.rb +1 -1
- data/app/models/spree/option_value.rb +11 -0
- data/app/models/spree/option_value_variant.rb +1 -1
- data/app/models/spree/order.rb +46 -45
- data/app/models/spree/order/address_book.rb +3 -5
- data/app/models/spree/order/currency_updater.rb +1 -1
- data/app/models/spree/order/emails.rb +32 -0
- data/app/models/spree/order/payments.rb +1 -1
- data/app/models/spree/order/store_credit.rb +1 -1
- data/app/models/spree/payment.rb +7 -0
- data/app/models/spree/payment_method.rb +4 -0
- data/app/models/spree/preferences/preferable.rb +12 -0
- data/app/models/spree/preferences/preferable_class_methods.rb +10 -1
- data/app/models/spree/product.rb +27 -22
- data/app/models/spree/product_property.rb +19 -4
- data/app/models/spree/promotion.rb +6 -15
- data/app/models/spree/promotion/rules/country.rb +1 -1
- data/app/models/spree/promotion/rules/first_order.rb +4 -3
- data/app/models/spree/promotion/rules/taxon.rb +10 -7
- data/app/models/spree/promotion_handler/cart.rb +7 -2
- data/app/models/spree/promotion_handler/coupon.rb +5 -4
- data/app/models/spree/promotion_handler/free_shipping.rb +5 -6
- data/app/models/spree/promotion_handler/page.rb +3 -2
- data/app/models/spree/promotion_handler/promotion_duplicator.rb +1 -0
- data/app/models/spree/promotion_rule.rb +2 -0
- data/app/models/spree/property.rb +27 -0
- data/app/models/spree/reimbursement.rb +3 -1
- data/app/models/spree/reimbursement_type/reimbursement_helpers.rb +2 -1
- data/app/models/spree/shipment_handler.rb +4 -2
- data/app/models/spree/stock/quantifier.rb +6 -6
- data/app/models/spree/store.rb +90 -7
- data/app/models/spree/store_credit.rb +7 -3
- data/app/models/spree/store_credit_event.rb +2 -2
- data/app/models/spree/store_payment_method.rb +11 -0
- data/app/models/spree/store_product.rb +11 -0
- data/app/models/spree/store_promotion.rb +11 -0
- data/app/models/spree/taxon.rb +21 -1
- data/app/models/spree/taxon_image/configuration/active_storage.rb +2 -0
- data/app/models/spree/taxonomy.rb +3 -1
- data/app/models/spree/variant.rb +13 -4
- data/app/presenters/spree/filters/options_presenter.rb +27 -0
- data/app/presenters/spree/filters/price_presenter.rb +22 -0
- data/app/presenters/spree/filters/price_range_presenter.rb +29 -0
- data/app/presenters/spree/filters/properties_presenter.rb +23 -0
- data/app/presenters/spree/filters/property_presenter.rb +22 -0
- data/app/presenters/spree/filters/quantified_price_range_presenter.rb +44 -0
- data/app/services/spree/account/addresses/create.rb +1 -0
- data/app/services/spree/account/addresses/helper.rb +6 -0
- data/app/services/spree/account/addresses/update.rb +9 -3
- data/app/services/spree/account/create.rb +17 -0
- data/app/services/spree/account/update.rb +15 -0
- data/app/services/spree/build_localized_redirect_url.rb +1 -1
- data/app/services/spree/cart/create.rb +5 -3
- data/app/services/spree/cart/destroy.rb +40 -0
- data/app/services/spree/cart/empty.rb +36 -0
- data/app/services/spree/cart/estimate_shipping_rates.rb +3 -3
- data/app/services/spree/checkout/add_store_credit.rb +1 -1
- data/app/services/spree/classifications/reposition.rb +18 -0
- data/app/services/spree/credit_cards/destroy.rb +41 -0
- data/app/validators/db_maximum_length_validator.rb +5 -0
- data/app/validators/email_validator.rb +3 -1
- data/app/views/spree/shared/_purchased_items_table.text.erb +25 -0
- data/config/initializers/active_storage.rb +2 -0
- data/config/locales/en.yml +166 -2
- data/config/routes.rb +0 -5
- data/db/migrate/20201023152810_add_filterable_to_spree_properties.rb +8 -0
- data/db/migrate/20210407200948_create_spree_menus.rb +16 -0
- data/db/migrate/20210408092939_create_spree_menu_items.rb +31 -0
- data/db/migrate/20210504163720_add_filter_param_to_spree_product_properties.rb +8 -0
- data/db/migrate/20210505114659_add_filter_param_to_spree_properties.rb +8 -0
- data/db/migrate/20210512191732_create_spree_cms_pages.rb +24 -0
- data/db/migrate/20210514204251_create_spree_cms_sections.rb +22 -0
- data/db/migrate/20210527094055_create_spree_products_stores.rb +29 -0
- data/db/migrate/20210608045519_ensure_store_default_country_is_set.rb +5 -0
- data/db/migrate/20210702112334_add_missing_timestamp_columns.rb +46 -0
- data/db/migrate/20210713131614_add_unique_index_on_property_id_and_product_id_to_product_properties.rb +29 -0
- data/db/migrate/20210715091956_add_store_id_to_spree_store_credits.rb +10 -0
- data/db/migrate/20210716093151_add_store_id_to_spree_taxonomies.rb +11 -0
- data/db/migrate/20210716104141_add_index_on_name_parent_id_and_taxonomy_id_on_spree_taxons.rb +31 -0
- data/db/migrate/20210721120857_add_index_on_permalink_parent_id_and_taxonomy_id_on_spree_taxons.rb +31 -0
- data/db/migrate/20210721125657_create_spree_promotions_stores.rb +29 -0
- data/db/migrate/20210722090705_add_store_id_to_spree_customer_returns.rb +11 -0
- data/db/migrate/20210726065456_change_integer_id_columns_into_bigint.rb +305 -0
- data/db/migrate/20210730154425_fix_promotion_code_and_path_unique_indexes.rb +9 -0
- data/lib/generators/spree/dummy/dummy_generator.rb +11 -6
- data/lib/generators/spree/dummy_model/templates/model.rb.tt +1 -1
- data/lib/generators/spree/install/install_generator.rb +12 -24
- data/lib/spree/core.rb +8 -4
- data/lib/spree/core/components.rb +8 -0
- data/lib/spree/core/controller_helpers/auth.rb +7 -1
- data/lib/spree/core/controller_helpers/common.rb +1 -1
- data/lib/spree/core/controller_helpers/order.rb +4 -4
- data/lib/spree/core/controller_helpers/search.rb +1 -0
- data/lib/spree/core/controller_helpers/store.rb +10 -1
- data/lib/spree/core/engine.rb +6 -0
- data/lib/spree/core/importer/product.rb +3 -1
- data/lib/spree/core/product_duplicator.rb +1 -0
- data/lib/spree/core/search/base.rb +17 -22
- data/lib/spree/core/version.rb +1 -1
- data/lib/spree/i18n.rb +1 -1
- data/lib/spree/money.rb +2 -1
- data/lib/spree/permitted_attributes.rb +14 -3
- data/lib/spree/testing_support/authorization_helpers.rb +6 -3
- data/lib/spree/testing_support/capybara_config.rb +3 -1
- data/lib/spree/testing_support/common_rake.rb +17 -4
- data/lib/spree/testing_support/extension_rake.rb +2 -2
- data/lib/spree/testing_support/factories/address_factory.rb +1 -1
- data/lib/spree/testing_support/factories/classification_factory.rb +8 -0
- data/lib/spree/testing_support/factories/cms_page_factory.rb +20 -0
- data/lib/spree/testing_support/factories/cms_section_factory.rb +31 -0
- data/lib/spree/testing_support/factories/customer_return_factory.rb +24 -17
- data/lib/spree/testing_support/factories/icon_factory.rb +7 -0
- data/lib/spree/testing_support/factories/line_item_factory.rb +6 -2
- data/lib/spree/testing_support/factories/menu_factory.rb +16 -0
- data/lib/spree/testing_support/factories/menu_item_factory.rb +10 -0
- data/lib/spree/testing_support/factories/options_factory.rb +5 -0
- data/lib/spree/testing_support/factories/order_factory.rb +11 -2
- data/lib/spree/testing_support/factories/payment_method_factory.rb +6 -2
- data/lib/spree/testing_support/factories/product_factory.rb +11 -1
- data/lib/spree/testing_support/factories/product_property_factory.rb +1 -1
- data/lib/spree/testing_support/factories/promotion_factory.rb +18 -9
- data/lib/spree/testing_support/factories/property_factory.rb +22 -0
- data/lib/spree/testing_support/factories/stock_location_factory.rb +5 -4
- data/lib/spree/testing_support/factories/store_credit_factory.rb +1 -0
- data/lib/spree/testing_support/factories/store_factory.rb +11 -0
- data/lib/spree/testing_support/factories/taxon_factory.rb +3 -1
- data/lib/spree/testing_support/factories/taxon_image_factory.rb +7 -0
- data/lib/spree/testing_support/factories/taxonomy_factory.rb +1 -0
- data/lib/spree/testing_support/factories/user_factory.rb +7 -2
- data/lib/spree/testing_support/factories/variant_factory.rb +2 -2
- data/lib/spree/testing_support/flatpickr_capybara.rb +58 -35
- data/lib/spree/testing_support/order_walkthrough.rb +9 -9
- data/lib/tasks/core.rake +1 -1
- data/spec/fixtures/favicon.ico +0 -0
- data/spec/fixtures/files/icon_256x256.gif +0 -0
- data/spec/fixtures/files/icon_256x256.png +0 -0
- data/spec/fixtures/files/icon_512x512.png +0 -0
- data/spec/fixtures/files/img_256x128.png +0 -0
- data/spree_core.gemspec +11 -10
- metadata +206 -157
- data/app/assets/images/logo/spree_50.png +0 -0
- data/app/assets/images/noimage/large.png +0 -0
- data/app/assets/images/noimage/mini.png +0 -0
- data/app/assets/images/noimage/product.png +0 -0
- data/app/assets/images/noimage/small.png +0 -0
- data/app/assets/javascripts/spree.js +0 -78
- data/app/controllers/spree/errors_controller.rb +0 -11
- data/app/helpers/spree/mail_helper.rb +0 -29
- data/app/mailers/spree/base_mailer.rb +0 -46
- data/app/mailers/spree/order_mailer.rb +0 -26
- data/app/mailers/spree/reimbursement_mailer.rb +0 -12
- data/app/mailers/spree/shipment_mailer.rb +0 -12
- data/app/mailers/spree/test_mailer.rb +0 -8
- data/app/models/spree/validations/db_maximum_length_validator.rb +0 -22
- data/app/views/layouts/spree/base_mailer.html.erb +0 -46
- data/app/views/spree/errors/forbidden.html.erb +0 -0
- data/app/views/spree/errors/unauthorized.html.erb +0 -0
- data/app/views/spree/order_mailer/cancel_email.html.erb +0 -24
- data/app/views/spree/order_mailer/cancel_email.text.erb +0 -38
- data/app/views/spree/order_mailer/confirm_email.html.erb +0 -23
- data/app/views/spree/order_mailer/confirm_email.text.erb +0 -39
- data/app/views/spree/order_mailer/store_owner_notification_email.html.erb +0 -23
- data/app/views/spree/order_mailer/store_owner_notification_email.text.erb +0 -38
- data/app/views/spree/reimbursement_mailer/reimbursement_email.html.erb +0 -56
- data/app/views/spree/reimbursement_mailer/reimbursement_email.text.erb +0 -24
- data/app/views/spree/shared/_base_mailer_footer.html.erb +0 -12
- data/app/views/spree/shared/_base_mailer_header.html.erb +0 -13
- data/app/views/spree/shared/_base_mailer_stylesheets.html.erb +0 -456
- data/app/views/spree/shared/_error_messages.html.erb +0 -11
- data/app/views/spree/shared/_mailer_line_item.html.erb +0 -16
- data/app/views/spree/shared/_paths.html.erb +0 -8
- data/app/views/spree/shared/_purchased_items_table.html.erb +0 -69
- data/app/views/spree/shared/purchased_items_table/_adjustment.html.erb +0 -13
- data/app/views/spree/shared/purchased_items_table/_line_item.html.erb +0 -27
- data/app/views/spree/shared/purchased_items_table/_subtotal.html.erb +0 -13
- data/app/views/spree/shared/purchased_items_table/_total.html.erb +0 -13
- data/app/views/spree/shipment_mailer/shipped_email.html.erb +0 -36
- data/app/views/spree/shipment_mailer/shipped_email.text.erb +0 -17
- data/app/views/spree/test_mailer/test_email.html.erb +0 -40
- data/app/views/spree/test_mailer/test_email.text.erb +0 -4
- data/config/initializers/assets.rb +0 -2
- data/config/initializers/premailer_assets.rb +0 -1
- data/config/initializers/premailer_rails.rb +0 -3
- data/db/migrate/20140805171219_make_existing_credit_cards_default.rb +0 -10
- data/lib/generators/spree/dummy/templates/initializers/bullet.rb +0 -5
- data/lib/generators/spree/install/templates/vendor/assets/javascripts/spree/backend/all.js +0 -15
- data/lib/generators/spree/install/templates/vendor/assets/javascripts/spree/frontend/all.js +0 -16
- data/lib/generators/spree/install/templates/vendor/assets/stylesheets/spree/backend/all.css +0 -16
- data/lib/generators/spree/install/templates/vendor/assets/stylesheets/spree/frontend/all.css +0 -16
- data/lib/generators/spree/mailers_preview/mailers_preview_generator.rb +0 -23
- data/lib/generators/spree/mailers_preview/templates/mailers/previews/order_preview.rb +0 -13
- data/lib/generators/spree/mailers_preview/templates/mailers/previews/reimbursement_preview.rb +0 -5
- data/lib/generators/spree/mailers_preview/templates/mailers/previews/shipment_preview.rb +0 -5
- data/lib/generators/spree/mailers_preview/templates/mailers/previews/user_preview.rb +0 -11
- data/lib/tasks/email.rake +0 -10
- data/vendor/assets/javascripts/cleave.js +0 -1669
- data/vendor/assets/javascripts/fetch.umd.js +0 -531
- data/vendor/assets/javascripts/jquery.payment.js +0 -652
- data/vendor/assets/javascripts/jsuri.js +0 -458
- data/vendor/assets/javascripts/polyfill.min.js +0 -1
@@ -1,5 +1,7 @@
|
|
1
1
|
module Spree
|
2
2
|
class StoreCredit < Spree::Base
|
3
|
+
include SingleStoreResource
|
4
|
+
|
3
5
|
acts_as_paranoid
|
4
6
|
|
5
7
|
VOID_ACTION = 'void'.freeze
|
@@ -16,9 +18,10 @@ module Spree
|
|
16
18
|
belongs_to :category, class_name: 'Spree::StoreCreditCategory'
|
17
19
|
belongs_to :created_by, class_name: Spree.user_class.to_s, foreign_key: 'created_by_id'
|
18
20
|
belongs_to :credit_type, class_name: 'Spree::StoreCreditType', foreign_key: 'type_id'
|
19
|
-
|
21
|
+
belongs_to :store, class_name: 'Spree::Store'
|
22
|
+
has_many :store_credit_events, class_name: 'Spree::StoreCreditEvent'
|
20
23
|
|
21
|
-
validates :user, :category, :credit_type, :created_by, :currency, presence: true
|
24
|
+
validates :user, :category, :credit_type, :created_by, :currency, :store, presence: true
|
22
25
|
validates :amount, numericality: { greater_than: 0 }
|
23
26
|
validates :amount_used, numericality: { greater_than_or_equal_to: 0 }
|
24
27
|
validate :amount_used_less_than_or_equal_to_amount
|
@@ -188,7 +191,8 @@ module Spree
|
|
188
191
|
created_by_id: created_by_id,
|
189
192
|
currency: currency,
|
190
193
|
type_id: type_id,
|
191
|
-
memo: credit_allocation_memo
|
194
|
+
memo: credit_allocation_memo,
|
195
|
+
store: store
|
192
196
|
}
|
193
197
|
end
|
194
198
|
|
@@ -8,7 +8,7 @@ module Spree
|
|
8
8
|
scope :exposed_events, -> { where.not(action: [Spree::StoreCredit::ELIGIBLE_ACTION, Spree::StoreCredit::AUTHORIZE_ACTION]) }
|
9
9
|
scope :reverse_chronological, -> { order(created_at: :desc) }
|
10
10
|
|
11
|
-
delegate :currency, to: :store_credit
|
11
|
+
delegate :currency, :store, to: :store_credit
|
12
12
|
|
13
13
|
def display_amount
|
14
14
|
Spree::Money.new(amount, currency: currency)
|
@@ -32,7 +32,7 @@ module Spree
|
|
32
32
|
end
|
33
33
|
|
34
34
|
def order
|
35
|
-
|
35
|
+
store.payments.find_by(response_code: authorization_code).try(:order)
|
36
36
|
end
|
37
37
|
end
|
38
38
|
end
|
@@ -0,0 +1,11 @@
|
|
1
|
+
module Spree
|
2
|
+
class StorePaymentMethod < Spree::Base
|
3
|
+
self.table_name = 'spree_payment_methods_stores'
|
4
|
+
|
5
|
+
belongs_to :store, class_name: 'Spree::Store', touch: true
|
6
|
+
belongs_to :payment_method, class_name: 'Spree::PaymentMethod', touch: true
|
7
|
+
|
8
|
+
validates :store, :payment_method, presence: true
|
9
|
+
validates :store_id, uniqueness: { scope: :payment_method_id }
|
10
|
+
end
|
11
|
+
end
|
@@ -0,0 +1,11 @@
|
|
1
|
+
module Spree
|
2
|
+
class StoreProduct < Spree::Base
|
3
|
+
self.table_name = 'spree_products_stores'
|
4
|
+
|
5
|
+
belongs_to :store, class_name: 'Spree::Store', touch: true
|
6
|
+
belongs_to :product, class_name: 'Spree::Product', touch: true
|
7
|
+
|
8
|
+
validates :store, :product, presence: true
|
9
|
+
validates :store_id, uniqueness: { scope: :product_id }
|
10
|
+
end
|
11
|
+
end
|
@@ -0,0 +1,11 @@
|
|
1
|
+
module Spree
|
2
|
+
class StorePromotion < Spree::Base
|
3
|
+
self.table_name = 'spree_promotions_stores'
|
4
|
+
|
5
|
+
belongs_to :store, class_name: 'Spree::Store', touch: true
|
6
|
+
belongs_to :promotion, class_name: 'Spree::Promotion', touch: true
|
7
|
+
|
8
|
+
validates :store, :promotion, presence: true
|
9
|
+
validates :store_id, uniqueness: { scope: :promotion_id }
|
10
|
+
end
|
11
|
+
end
|
data/app/models/spree/taxon.rb
CHANGED
@@ -13,6 +13,9 @@ 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_many :menu_items, as: :linked_resource
|
17
|
+
has_many :cms_sections, as: :linked_resource
|
18
|
+
|
16
19
|
has_many :prototype_taxons, class_name: 'Spree::PrototypeTaxon', dependent: :destroy
|
17
20
|
has_many :prototypes, through: :prototype_taxons, class_name: 'Spree::Prototype'
|
18
21
|
|
@@ -20,23 +23,30 @@ module Spree
|
|
20
23
|
has_many :promotion_rules, through: :promotion_rule_taxons, class_name: 'Spree::PromotionRule'
|
21
24
|
|
22
25
|
validates :name, presence: true, uniqueness: { scope: [:parent_id, :taxonomy_id], allow_blank: true, case_sensitive: false }
|
23
|
-
validates :
|
26
|
+
validates :taxonomy, presence: true
|
27
|
+
validates :permalink, uniqueness: { case_sensitive: false, scope: [:parent_id, :taxonomy_id] }
|
24
28
|
validates :hide_from_nav, inclusion: { in: [true, false] }
|
25
29
|
validates_associated :icon
|
26
30
|
validate :check_for_root, on: :create
|
31
|
+
validate :parent_belongs_to_same_taxonomy
|
27
32
|
with_options length: { maximum: 255 }, allow_blank: true do
|
28
33
|
validates :meta_keywords
|
29
34
|
validates :meta_description
|
30
35
|
validates :meta_title
|
31
36
|
end
|
32
37
|
|
38
|
+
before_validation :copy_taxonomy_from_parent
|
33
39
|
after_save :touch_ancestors_and_taxonomy
|
34
40
|
after_touch :touch_ancestors_and_taxonomy
|
35
41
|
|
36
42
|
has_one :icon, as: :viewable, dependent: :destroy, class_name: 'Spree::TaxonImage'
|
37
43
|
|
44
|
+
scope :for_store, ->(store) { joins(:taxonomy).where(spree_taxonomies: { store_id: store.id }) }
|
45
|
+
|
38
46
|
self.whitelisted_ransackable_associations = %w[taxonomy]
|
39
47
|
|
48
|
+
scope :for_stores, ->(stores) { joins(:taxonomy).where(spree_taxonomies: { store_id: stores.ids }) }
|
49
|
+
|
40
50
|
# indicate which filters should be used for a taxon
|
41
51
|
# this method should be customized to your own site
|
42
52
|
def applicable_filters
|
@@ -104,5 +114,15 @@ module Spree
|
|
104
114
|
errors.add(:root_conflict, 'this taxonomy already has a root taxon')
|
105
115
|
end
|
106
116
|
end
|
117
|
+
|
118
|
+
def parent_belongs_to_same_taxonomy
|
119
|
+
if parent.present? && parent.taxonomy_id != taxonomy_id
|
120
|
+
errors.add(:parent, 'must belong to the same taxonomy')
|
121
|
+
end
|
122
|
+
end
|
123
|
+
|
124
|
+
def copy_taxonomy_from_parent
|
125
|
+
self.taxonomy = parent.taxonomy if parent.present? && taxonomy.blank?
|
126
|
+
end
|
107
127
|
end
|
108
128
|
end
|
@@ -2,10 +2,12 @@ module Spree
|
|
2
2
|
class Taxonomy < Spree::Base
|
3
3
|
acts_as_list
|
4
4
|
|
5
|
-
validates :name, presence: true, uniqueness: { case_sensitive: false, allow_blank: true }
|
5
|
+
validates :name, presence: true, uniqueness: { case_sensitive: false, allow_blank: true, scope: :store_id }
|
6
|
+
validates :store, presence: true
|
6
7
|
|
7
8
|
has_many :taxons, inverse_of: :taxonomy
|
8
9
|
has_one :root, -> { where parent_id: nil }, class_name: 'Spree::Taxon', dependent: :destroy
|
10
|
+
belongs_to :store, class_name: 'Spree::Store'
|
9
11
|
|
10
12
|
after_create :set_root
|
11
13
|
after_save :set_root_taxon_name
|
data/app/models/spree/variant.rb
CHANGED
@@ -3,6 +3,10 @@ module Spree
|
|
3
3
|
acts_as_paranoid
|
4
4
|
acts_as_list scope: :product
|
5
5
|
|
6
|
+
include MemoizedData
|
7
|
+
|
8
|
+
MEMOIZED_METHODS = %w(purchasable in_stock backorderable tax_category options_text compare_at_price)
|
9
|
+
|
6
10
|
belongs_to :product, -> { with_deleted }, touch: true, class_name: 'Spree::Product', inverse_of: :variants
|
7
11
|
belongs_to :tax_category, class_name: 'Spree::TaxCategory', optional: true
|
8
12
|
|
@@ -29,7 +33,7 @@ module Spree
|
|
29
33
|
end
|
30
34
|
|
31
35
|
has_many :option_value_variants, class_name: 'Spree::OptionValueVariant'
|
32
|
-
has_many :option_values, through: :option_value_variants, class_name: 'Spree::OptionValue'
|
36
|
+
has_many :option_values, through: :option_value_variants, dependent: :destroy, class_name: 'Spree::OptionValue'
|
33
37
|
|
34
38
|
has_many :images, -> { order(:position) }, as: :viewable, dependent: :destroy, class_name: 'Spree::Image'
|
35
39
|
|
@@ -93,6 +97,7 @@ module Spree
|
|
93
97
|
not_discontinued.not_deleted.
|
94
98
|
for_currency_and_available_price_amount(currency)
|
95
99
|
end
|
100
|
+
# FIXME: cost price should be represented with DisplayMoney class
|
96
101
|
LOCALIZED_NUMBERS = %w(cost_price weight depth width height)
|
97
102
|
|
98
103
|
LOCALIZED_NUMBERS.each do |m|
|
@@ -199,6 +204,10 @@ module Spree
|
|
199
204
|
price_in(currency).try(:amount)
|
200
205
|
end
|
201
206
|
|
207
|
+
def compare_at_amount_in(currency)
|
208
|
+
price_in(currency).try(:compare_at_amount)
|
209
|
+
end
|
210
|
+
|
202
211
|
def price_modifier_amount_in(currency, options = {})
|
203
212
|
return 0 unless options.present?
|
204
213
|
|
@@ -242,7 +251,7 @@ module Spree
|
|
242
251
|
# Check if model responds to cache version and fall back to updated_at for older rails versions
|
243
252
|
# This makes sure a version is supplied when recyclable cache keys are disabled.
|
244
253
|
version = respond_to?(:cache_version) ? cache_version : updated_at.to_i
|
245
|
-
Rails.cache.fetch(in_stock_cache_key, version: version) do
|
254
|
+
@in_stock ||= Rails.cache.fetch(in_stock_cache_key, version: version) do
|
246
255
|
total_on_hand > 0
|
247
256
|
end
|
248
257
|
end
|
@@ -252,7 +261,7 @@ module Spree
|
|
252
261
|
alias is_backorderable? backorderable?
|
253
262
|
|
254
263
|
def purchasable?
|
255
|
-
in_stock? || backorderable?
|
264
|
+
@purchasable ||= in_stock? || backorderable?
|
256
265
|
end
|
257
266
|
|
258
267
|
# Shortcut method to determine if inventory tracking is enabled for this variant
|
@@ -282,7 +291,7 @@ module Spree
|
|
282
291
|
end
|
283
292
|
|
284
293
|
def backordered?
|
285
|
-
total_on_hand <= 0 && stock_items.exists?(backorderable: true)
|
294
|
+
@backordered ||= total_on_hand <= 0 && stock_items.exists?(backorderable: true)
|
286
295
|
end
|
287
296
|
|
288
297
|
private
|
@@ -0,0 +1,27 @@
|
|
1
|
+
module Spree
|
2
|
+
module Filters
|
3
|
+
class OptionsPresenter
|
4
|
+
FilterableOptionType = Struct.new(:option_type, :option_values, keyword_init: true) do
|
5
|
+
delegate_missing_to :option_type
|
6
|
+
end
|
7
|
+
|
8
|
+
def initialize(option_values_scope:)
|
9
|
+
@option_values = option_values_scope.includes(:option_type)
|
10
|
+
end
|
11
|
+
|
12
|
+
def to_a
|
13
|
+
grouped_options.map do |option_type, option_values|
|
14
|
+
FilterableOptionType.new(option_type: option_type, option_values: option_values)
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
private
|
19
|
+
|
20
|
+
attr_reader :option_values
|
21
|
+
|
22
|
+
def grouped_options
|
23
|
+
option_values.group_by(&:option_type)
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
module Spree
|
2
|
+
module Filters
|
3
|
+
class PricePresenter
|
4
|
+
def initialize(amount:, currency:)
|
5
|
+
@amount = amount.to_i
|
6
|
+
@currency = currency
|
7
|
+
end
|
8
|
+
|
9
|
+
def to_i
|
10
|
+
amount
|
11
|
+
end
|
12
|
+
|
13
|
+
def to_s
|
14
|
+
Spree::Money.new(amount, currency: currency, no_cents_if_whole: true).to_s
|
15
|
+
end
|
16
|
+
|
17
|
+
private
|
18
|
+
|
19
|
+
attr_reader :amount, :currency
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
module Spree
|
2
|
+
module Filters
|
3
|
+
class PriceRangePresenter
|
4
|
+
def self.from_param(param, currency:)
|
5
|
+
prices = param.split('-')
|
6
|
+
|
7
|
+
new(
|
8
|
+
min_price: PricePresenter.new(amount: prices.first, currency: currency),
|
9
|
+
max_price: PricePresenter.new(amount: prices.last, currency: currency)
|
10
|
+
)
|
11
|
+
end
|
12
|
+
|
13
|
+
def initialize(min_price:, max_price:)
|
14
|
+
@min_price = min_price
|
15
|
+
@max_price = max_price
|
16
|
+
end
|
17
|
+
|
18
|
+
attr_reader :min_price, :max_price
|
19
|
+
|
20
|
+
def to_param
|
21
|
+
"#{min_price.to_i}-#{max_price.to_i}"
|
22
|
+
end
|
23
|
+
|
24
|
+
def to_s
|
25
|
+
"#{min_price} - #{max_price}"
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
module Spree
|
2
|
+
module Filters
|
3
|
+
class PropertiesPresenter
|
4
|
+
def initialize(product_properties_scope:)
|
5
|
+
@product_properties = product_properties_scope.includes(:property)
|
6
|
+
end
|
7
|
+
|
8
|
+
def to_a
|
9
|
+
grouped_options.map do |property, product_properties|
|
10
|
+
PropertyPresenter.new(property: property, product_properties: product_properties)
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
private
|
15
|
+
|
16
|
+
attr_reader :product_properties
|
17
|
+
|
18
|
+
def grouped_options
|
19
|
+
product_properties.group_by(&:property)
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
module Spree
|
2
|
+
module Filters
|
3
|
+
class PropertyPresenter
|
4
|
+
def initialize(property:, product_properties:)
|
5
|
+
@property = property
|
6
|
+
@product_properties = product_properties
|
7
|
+
end
|
8
|
+
|
9
|
+
attr_reader :product_properties
|
10
|
+
|
11
|
+
delegate_missing_to :property
|
12
|
+
|
13
|
+
def uniq_values
|
14
|
+
property.uniq_values(product_properties_scope: product_properties)
|
15
|
+
end
|
16
|
+
|
17
|
+
private
|
18
|
+
|
19
|
+
attr_reader :property
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,44 @@
|
|
1
|
+
module Spree
|
2
|
+
module Filters
|
3
|
+
class QuantifiedPriceRangePresenter
|
4
|
+
ALLOWED_QUANTIFIERS = [
|
5
|
+
:less_than,
|
6
|
+
:more_than
|
7
|
+
].freeze
|
8
|
+
|
9
|
+
def initialize(price:, quantifier:)
|
10
|
+
if ALLOWED_QUANTIFIERS.exclude?(quantifier.to_sym)
|
11
|
+
raise ArgumentError, "quantifier must be one of: #{ALLOWED_QUANTIFIERS.join(', ')}"
|
12
|
+
end
|
13
|
+
|
14
|
+
@price = price
|
15
|
+
@quantifier = quantifier.to_sym
|
16
|
+
end
|
17
|
+
|
18
|
+
def to_param
|
19
|
+
case quantifier
|
20
|
+
when :less_than
|
21
|
+
less_than_param
|
22
|
+
when :more_than
|
23
|
+
more_than_param
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
def to_s
|
28
|
+
"#{I18n.t("activerecord.attributes.spree/product.#{quantifier}")} #{price}"
|
29
|
+
end
|
30
|
+
|
31
|
+
private
|
32
|
+
|
33
|
+
attr_reader :price, :quantifier
|
34
|
+
|
35
|
+
def less_than_param
|
36
|
+
"0-#{price.to_i}"
|
37
|
+
end
|
38
|
+
|
39
|
+
def more_than_param
|
40
|
+
"#{price.to_i}-Infinity"
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
@@ -31,6 +31,12 @@ module Spree
|
|
31
31
|
params[:state_id] = country.states.find_by(name: state_name)&.id
|
32
32
|
params
|
33
33
|
end
|
34
|
+
|
35
|
+
def assign_to_user_as_default(user:, address_id:)
|
36
|
+
if user.addresses.pluck(:id) == [address_id] # check if it's user first address
|
37
|
+
user.update(bill_address_id: address_id, ship_address_id: address_id)
|
38
|
+
end
|
39
|
+
end
|
34
40
|
end
|
35
41
|
end
|
36
42
|
end
|
@@ -11,12 +11,18 @@ module Spree
|
|
11
11
|
address_params[:country_id] ||= address.country_id
|
12
12
|
fill_country_and_state_ids(address_params)
|
13
13
|
|
14
|
-
if address
|
15
|
-
success(address)
|
14
|
+
if address&.editable?
|
15
|
+
address.update(address_params) ? success(address) : failure(address)
|
16
16
|
else
|
17
|
-
|
17
|
+
new_address(address_params).valid? ? address.destroy && success(new_address) : failure(new_address)
|
18
18
|
end
|
19
19
|
end
|
20
|
+
|
21
|
+
private
|
22
|
+
|
23
|
+
def new_address(address_params = {})
|
24
|
+
@new_address ||= ::Spree::Address.find_or_create_by(address_params.except(:id, :updated_at, :created_at))
|
25
|
+
end
|
20
26
|
end
|
21
27
|
end
|
22
28
|
end
|