From 3d957d31791da07a0ecbc7feac467499623a60d1 Mon Sep 17 00:00:00 2001 From: Tomasz Donarski Date: Mon, 13 Nov 2023 08:46:49 +0100 Subject: [PATCH 1/4] Initial implementation of bulk promo codes admin UI --- .../admin/promotion_batches_controller.rb | 63 +++++++++++++++++ .../spree/admin/promotions_controller.rb | 1 + app/helpers/spree/admin/navigation_helper.rb | 8 +++ ...promotion_batch_default_actions_builder.rb | 27 ++++++++ ...omotion_batches_default_actions_builder.rb | 28 ++++++++ .../default_configuration_builder.rb | 8 +++ .../admin/promotion_batches/_form.html.erb | 6 ++ .../admin/promotion_batches/_size.html.erb | 6 ++ .../admin/promotion_batches/edit.html.erb | 68 +++++++++++++++++++ .../admin/promotion_batches/index.html.erb | 45 ++++++++++++ .../admin/promotion_batches/new.html.erb | 13 ++++ .../admin/promotion_batches/show.html.erb | 37 ++++++++++ config/routes.rb | 8 +++ lib/spree/backend/engine.rb | 4 ++ ...tion_batch_default_actions_builder_spec.rb | 20 ++++++ ...on_batches_default_actions_builder_spec.rb | 20 ++++++ .../default_configuration_builder_spec.rb | 2 +- 17 files changed, 363 insertions(+), 1 deletion(-) create mode 100644 app/controllers/spree/admin/promotion_batches_controller.rb create mode 100644 app/models/spree/admin/actions/promotion_batch_default_actions_builder.rb create mode 100644 app/models/spree/admin/actions/promotion_batches_default_actions_builder.rb create mode 100644 app/views/spree/admin/promotion_batches/_form.html.erb create mode 100644 app/views/spree/admin/promotion_batches/_size.html.erb create mode 100644 app/views/spree/admin/promotion_batches/edit.html.erb create mode 100644 app/views/spree/admin/promotion_batches/index.html.erb create mode 100644 app/views/spree/admin/promotion_batches/new.html.erb create mode 100644 app/views/spree/admin/promotion_batches/show.html.erb create mode 100644 spec/models/spree/admin/actions/promotion_batch_default_actions_builder_spec.rb create mode 100644 spec/models/spree/admin/actions/promotion_batches_default_actions_builder_spec.rb diff --git a/app/controllers/spree/admin/promotion_batches_controller.rb b/app/controllers/spree/admin/promotion_batches_controller.rb new file mode 100644 index 0000000000..d60e3e83b5 --- /dev/null +++ b/app/controllers/spree/admin/promotion_batches_controller.rb @@ -0,0 +1,63 @@ +module Spree + module Admin + class PromotionBatchesController < ResourceController + def update + if @object.template_promotion_id + flash[:error] = Spree.t(:template_promotion_already_assigned) + respond_with(@object) do |format| + format.html { render action: :edit, status: :unprocessable_entity } + format.js { render layout: false, status: :unprocessable_entity } + end + return + end + super + end + + def destroy + result = Spree::PromotionBatches::Destroy.call(promotion_batch: @promotion_batch) + + if result.success? + flash[:success] = flash_message_for(@promotion_batch, :successfully_removed) + else + flash[:error] = @promotion_batch.errors.full_messages.join(', ') + end + + respond_with(@promotion_batch) do |format| + format.html { redirect_to location_after_destroy } + format.js { render_js_for_destroy } + end + end + + def csv_export + send_data Spree::PromotionBatches::PromotionCodesExporter.new(params).call, + filename: "promo_codes_from_batch_id_#{params[:id]}.csv", + disposition: :attachment, + type: 'text/csv' + end + + def csv_import + file = params[:file] + Spree::PromotionBatches::PromotionCodesImporter.new(file: file, promotion_batch_id: params[:id]).call + redirect_back fallback_location: admin_promotions_path, notice: Spree.t('code_upload') + rescue Spree::PromotionBatches::PromotionCodesImporter::Error => e + redirect_back fallback_location: admin_promotions_path, alert: e.message + end + + def populate + batch_id = params[:id] + options = { + batch_size: params[:batch_size].to_i, + affix: params.dig(:code, :affix)&.to_sym, + content: params[:affix_content], + deny_list: params[:forbidden_phrases].split, + random_part_bytes: params[:random_part_bytes].to_i + } + + Spree::Promotions::PopulatePromotionBatch.new(batch_id, options).call + + flash[:success] = Spree.t('promotion_batch_populated') + redirect_to spree.edit_admin_promotion_batch_url(@promotion_batch) + end + end + end +end diff --git a/app/controllers/spree/admin/promotions_controller.rb b/app/controllers/spree/admin/promotions_controller.rb index 23a616cc90..4b66698654 100644 --- a/app/controllers/spree/admin/promotions_controller.rb +++ b/app/controllers/spree/admin/promotions_controller.rb @@ -40,6 +40,7 @@ def collection params[:q][:s] ||= 'id desc' @collection = super + @collection = @collection.non_batched @search = @collection.ransack(params[:q]) @collection = @search.result(distinct: true). includes(promotion_includes). diff --git a/app/helpers/spree/admin/navigation_helper.rb b/app/helpers/spree/admin/navigation_helper.rb index dfecab836c..7b6c81a7b8 100644 --- a/app/helpers/spree/admin/navigation_helper.rb +++ b/app/helpers/spree/admin/navigation_helper.rb @@ -377,6 +377,14 @@ def variants_actions def product_properties_actions Rails.application.config.spree_backend.actions[:product_properties] end + + def promotion_batch_actions + Rails.application.config.spree_backend.actions[:promotion_batch_actions] + end + + def promotion_batches_actions + Rails.application.config.spree_backend.actions[:promotion_batches_actions] + end # rubocop:enable Metrics/ModuleLength end end diff --git a/app/models/spree/admin/actions/promotion_batch_default_actions_builder.rb b/app/models/spree/admin/actions/promotion_batch_default_actions_builder.rb new file mode 100644 index 0000000000..f9c7979b9d --- /dev/null +++ b/app/models/spree/admin/actions/promotion_batch_default_actions_builder.rb @@ -0,0 +1,27 @@ +module Spree + module Admin + module Actions + class PromotionBatchDefaultActionsBuilder + include Spree::Core::Engine.routes.url_helpers + + def build + root = Root.new + export_codes_to_csv_action(root) + root + end + + private + + def export_codes_to_csv_action(root) + action = + ActionBuilder.new('csv_export', ->(resource) { csv_export_admin_promotion_batch_path(resource) }). + with_icon_key('download.svg'). + with_style(::Spree::Admin::Actions::ActionStyle::PRIMARY). + build + + root.add(action) + end + end + end + end +end diff --git a/app/models/spree/admin/actions/promotion_batches_default_actions_builder.rb b/app/models/spree/admin/actions/promotion_batches_default_actions_builder.rb new file mode 100644 index 0000000000..79a6ea955f --- /dev/null +++ b/app/models/spree/admin/actions/promotion_batches_default_actions_builder.rb @@ -0,0 +1,28 @@ +module Spree + module Admin + module Actions + class PromotionBatchesDefaultActionsBuilder + include Spree::Core::Engine.routes.url_helpers + + def build + root = Root.new + add_new_promotion_batch_action(root) + root + end + + private + + def add_new_promotion_batch_action(root) + action = + ActionBuilder.new('new_promotion_batch', new_admin_promotion_batch_path). + with_icon_key('add.svg'). + with_style(::Spree::Admin::Actions::ActionStyle::PRIMARY). + with_create_ability_check(::Spree::PromotionBatch). + build + + root.add(action) + end + end + end + end +end diff --git a/app/models/spree/admin/main_menu/default_configuration_builder.rb b/app/models/spree/admin/main_menu/default_configuration_builder.rb index 4ad4887506..b524eaada3 100644 --- a/app/models/spree/admin/main_menu/default_configuration_builder.rb +++ b/app/models/spree/admin/main_menu/default_configuration_builder.rb @@ -19,6 +19,7 @@ def build add_integrations_section(root) add_oauth_section(root) add_settings_section(root) + add_promotion_batches_section(root) root end @@ -255,6 +256,13 @@ def add_settings_section(root) build root.add(section) end + + def add_promotion_batches_section(root) + root.add(ItemBuilder.new('promotion_batches', admin_promotion_batches_path). + with_icon_key('stack.svg'). + with_admin_ability_check(Spree::PromotionBatch). + build) + end # rubocop:enable Metrics/AbcSize end # rubocop:enable Metrics/ClassLength diff --git a/app/views/spree/admin/promotion_batches/_form.html.erb b/app/views/spree/admin/promotion_batches/_form.html.erb new file mode 100644 index 0000000000..c831fea953 --- /dev/null +++ b/app/views/spree/admin/promotion_batches/_form.html.erb @@ -0,0 +1,6 @@ +
+ <%= f.field_container :template_promotion do %> + <%= f.label :template_promotion_id, Spree.t(:template_promotion) %> + <%= f.collection_select :template_promotion_id, Spree::Promotion.non_batched, :id, ->(promotion) { "#{promotion.name} # #{promotion.id}" }, { include_blank: true }, { class: 'select2-clear w-100' } %> + <% end %> +
diff --git a/app/views/spree/admin/promotion_batches/_size.html.erb b/app/views/spree/admin/promotion_batches/_size.html.erb new file mode 100644 index 0000000000..ac444014fb --- /dev/null +++ b/app/views/spree/admin/promotion_batches/_size.html.erb @@ -0,0 +1,6 @@ +
+ <%= form_tag(duplicate_admin_promotion_batch_path(@promotion_batch), method: :post) do %> + <%= number_field_tag(:batch_size) %> + <%= submit_tag("Duplicate") %> + <% end %> +
\ No newline at end of file diff --git a/app/views/spree/admin/promotion_batches/edit.html.erb b/app/views/spree/admin/promotion_batches/edit.html.erb new file mode 100644 index 0000000000..8c1f21c7f7 --- /dev/null +++ b/app/views/spree/admin/promotion_batches/edit.html.erb @@ -0,0 +1,68 @@ +<% content_for :page_title do %> + <%= link_to Spree.t(:promotion_batches), admin_promotion_batches_path %> / + <%= @promotion_batch.id %> +<% end %> + +<% if @promotion_batch.template_promotion_id%> +
+
+ <%= form_tag csv_import_admin_promotion_batch_path, method: :post, multipart: true do %> + <%= file_field_tag :file, accept: ".csv" %> + <%= submit_tag Spree.t('promotion_batch_form.import_codes') %> + <% end %> +
+
+<% end %> + +<% unless @promotion_batch.template_promotion_id%> + <%= form_for @promotion_batch, url: object_url, method: :put do |f| %> +
+
+ <%= render partial: 'form', locals: { f: f } %> + <%= render partial: 'spree/admin/shared/edit_resource_links' %> +
+
+ <% end %> +<% end %> + +<% if @promotion_batch.template_promotion_id %> + <%= form_with url: populate_admin_promotion_batch_path(@promotion_batch), method: :post do %> +
+ <%= label_tag :batch_size do %> + <%= number_field_tag(:batch_size) %> + <%= Spree.t('promotion_batch_form.batch_size') %> + <% end %> +
+
+ <%= label_tag :prefix do %> + <%= radio_button :code, :affix, 'prefix' %> + <%= Spree.t('promotion_batch_form.prefix') %> + <% end %> +
+
+ <%= label_tag :suffix do %> + <%= radio_button :code, :affix, 'suffix' %> + <%= Spree.t('promotion_batch_form.suffix') %> + <% end %> +
+
+ <%= label_tag :affix_content do %> + <%= text_field_tag(:affix_content) %> + <%= Spree.t('promotion_batch_form.affix_content') %> + <% end %> +
+
+ <%= label_tag :forbidden_phrases do %> + <%= text_area_tag(:forbidden_phrases) %> + <%= Spree.t('promotion_batch_form.forbidden_phrases') %> + <% end %> +
+
+ <%= label_tag :random_part_bytes do %> + <%= number_field_tag(:random_part_bytes, value = 4) %> + <%= Spree.t('promotion_batch_form.random_part_bytes') %> + <% end %> +
+ <%= submit_tag(Spree.t('promotion_batch_form.populate')) %> + <% end %> +<% end %> diff --git a/app/views/spree/admin/promotion_batches/index.html.erb b/app/views/spree/admin/promotion_batches/index.html.erb new file mode 100644 index 0000000000..60b460ae49 --- /dev/null +++ b/app/views/spree/admin/promotion_batches/index.html.erb @@ -0,0 +1,45 @@ +<% content_for :page_title do %> + <%= plural_resource_name(Spree::PromotionBatch) %> +<% end %> + +<% content_for :page_actions do %> + <% promotion_batches_actions.items.each do |action| %> + <% next unless action.available?(current_ability) %> + <%= button_link_to( + Spree.t(action.label_translation_key), + action.url, + class: action.classes, + icon: action.icon_key + ) %> + <% end %> +<% end %> + +<% if @promotion_batches.any? %> +
+ + + + + + + + + + + <% @promotion_batches.each do |promotion_batch| %> + + + + + + + <% end %> + +
<%= Spree.t(:id) %><%= Spree.t(:size) %><%= Spree.t(:template_promotion) %>
<%= link_to Spree::PromotionBatchPresenter.new(promotion_batch).call[:model_name_id], spree.admin_promotion_batch_path(promotion_batch) %><%= promotion_batch.promotions.count %><%= link_to Spree::PromotionBatchPresenter.new(promotion_batch).call[:template_promotion_name_id], spree.edit_admin_promotion_path(promotion_batch.template_promotion) if promotion_batch.template_promotion %> + + <%= link_to_edit promotion_batch, no_text: true if can?(:edit, promotion_batch) %> + <%= link_to_delete promotion_batch, no_text: true if can?(:delete, promotion_batch) %> + +
+
+<% end %> \ No newline at end of file diff --git a/app/views/spree/admin/promotion_batches/new.html.erb b/app/views/spree/admin/promotion_batches/new.html.erb new file mode 100644 index 0000000000..328fe5e0c0 --- /dev/null +++ b/app/views/spree/admin/promotion_batches/new.html.erb @@ -0,0 +1,13 @@ +<% content_for :page_title do %> + <%= link_to Spree.t(:promotion_batches), admin_promotion_batches_path %> / + <%= Spree.t(:new_promotion_batch) %> +<% end %> + +
+
+ <%= form_for :promotion_batch, url: collection_url do |f| %> + <%= render partial: 'form', locals: { f: f } %> + <%= render partial: 'spree/admin/shared/new_resource_links' %> + <% end %> +
+
diff --git a/app/views/spree/admin/promotion_batches/show.html.erb b/app/views/spree/admin/promotion_batches/show.html.erb new file mode 100644 index 0000000000..2bad5bdcfe --- /dev/null +++ b/app/views/spree/admin/promotion_batches/show.html.erb @@ -0,0 +1,37 @@ +<% content_for :page_title do %> + <%= link_to Spree.t(:promotion_batches), admin_promotion_batches_path %> / + <%= @promotion_batch.id %> +<% end %> + +<% content_for :page_actions do %> + <% promotion_batch_actions.items.each do |action| %> + <% next unless action.available?(current_ability) %> + <%= button_link_to( + Spree.t(action.label_translation_key), + action.url(@promotion_batch), + class: action.classes, + icon: action.icon_key + ) %> + <% end %> +<% end %> + + + + + + + + + + + + <% @promotion_batch.promotions.each do |promotion| %> + + + + + + + <% end %> + +
<%= Spree.t(:code) %><%= Spree.t(:description) %><%= Spree.t(:redeemed) %><%= Spree.t(:expiration) %>
<%= promotion.code %><%= promotion.description %><%= promotion.credits_count == promotion.usage_limit ? Spree.t(:say_yes) : Spree.t(:say_no) %><%= promotion.expires_at.to_date if promotion.expires_at %>
diff --git a/config/routes.rb b/config/routes.rb index e1facaf504..c1ee28aa19 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -10,6 +10,14 @@ resources :promotion_categories, except: [:show] + resources :promotion_batches do + member do + get :csv_export, to: 'promotion_batches#csv_export' + post :csv_import, to: 'promotion_batches#csv_import' + post :populate + end + end + resources :zones resources :stores, except: %i[index show] do diff --git a/lib/spree/backend/engine.rb b/lib/spree/backend/engine.rb index 49887afbff..119358659a 100644 --- a/lib/spree/backend/engine.rb +++ b/lib/spree/backend/engine.rb @@ -48,6 +48,10 @@ class Engine < ::Rails::Engine Rails.application.config.spree_backend.actions[:payments] = Spree::Admin::Actions::PaymentsDefaultActionsBuilder.new.build Rails.application.config.spree_backend.actions[:variants] = Spree::Admin::Actions::VariantsDefaultActionsBuilder.new.build Rails.application.config.spree_backend.actions[:product_properties] = Spree::Admin::Actions::ProductPropertiesDefaultActionsBuilder.new.build + Rails.application.config.spree_backend.actions[:promotion_batch_actions] = + Spree::Admin::Actions::PromotionBatchDefaultActionsBuilder.new.build + Rails.application.config.spree_backend.actions[:promotion_batches_actions] = + Spree::Admin::Actions::PromotionBatchesDefaultActionsBuilder.new.build end end end diff --git a/spec/models/spree/admin/actions/promotion_batch_default_actions_builder_spec.rb b/spec/models/spree/admin/actions/promotion_batch_default_actions_builder_spec.rb new file mode 100644 index 0000000000..2f45ca3460 --- /dev/null +++ b/spec/models/spree/admin/actions/promotion_batch_default_actions_builder_spec.rb @@ -0,0 +1,20 @@ +require 'spec_helper' + +module Spree + module Admin + describe Actions::PromotionBatchDefaultActionsBuilder, type: :model do + let(:builder) { described_class.new } + let(:default_actions) do + %w(csv_export) + end + + describe '#build' do + subject { builder.build } + + it 'builds default tabs' do + expect(subject.items.map(&:key)).to match(default_actions) + end + end + end + end +end diff --git a/spec/models/spree/admin/actions/promotion_batches_default_actions_builder_spec.rb b/spec/models/spree/admin/actions/promotion_batches_default_actions_builder_spec.rb new file mode 100644 index 0000000000..848f284f15 --- /dev/null +++ b/spec/models/spree/admin/actions/promotion_batches_default_actions_builder_spec.rb @@ -0,0 +1,20 @@ +require 'spec_helper' + +module Spree + module Admin + describe Actions::PromotionBatchesDefaultActionsBuilder, type: :model do + let(:builder) { described_class.new } + let(:default_actions) do + %w(new_promotion_batch) + end + + describe '#build' do + subject { builder.build } + + it 'builds default tabs' do + expect(subject.items.map(&:key)).to match(default_actions) + end + end + end + end +end diff --git a/spec/models/spree/admin/main_menu/default_configuration_builder_spec.rb b/spec/models/spree/admin/main_menu/default_configuration_builder_spec.rb index 353f05a9dc..455a3fec23 100644 --- a/spec/models/spree/admin/main_menu/default_configuration_builder_spec.rb +++ b/spec/models/spree/admin/main_menu/default_configuration_builder_spec.rb @@ -9,7 +9,7 @@ module Admin subject { builder.build } it 'builds a valid menu' do - expect(subject.items.count).to eq(12) + expect(subject.items.count).to eq(13) expect(subject.items.map(&:key)).to include('dashboard') expect(subject.items.map(&:key)).to include('orders') expect(subject.items.map(&:key)).to include('settings') From 70e43f007ea13ea20e56609cab4eb8206c5b6276 Mon Sep 17 00:00:00 2001 From: Rafal Cymerys Date: Wed, 14 Feb 2024 15:21:02 +0100 Subject: [PATCH 2/4] Cleanup admin panel implementation for bulk promo codes --- .../admin/promotion_batches_controller.rb | 96 ++++++++----------- .../spree/admin/promotions_controller.rb | 3 +- .../admin/template_promotions_controller.rb | 82 ++++++++++++++++ app/helpers/spree/admin/navigation_helper.rb | 6 +- ...promotion_batch_default_actions_builder.rb | 27 ------ ...omotion_batches_default_actions_builder.rb | 37 ++++++- ...plate_promotion_default_actions_builder.rb | 55 +++++++++++ .../default_configuration_builder.rb | 12 +-- .../admin/promotion_batches/_form.html.erb | 30 +++++- .../admin/promotion_batches/edit.html.erb | 68 ------------- .../admin/promotion_batches/import.html.erb | 24 +++++ .../admin/promotion_batches/index.html.erb | 33 ++++--- .../admin/promotion_batches/new.html.erb | 6 +- .../admin/promotion_batches/show.html.erb | 37 ------- .../admin/template_promotions/_form.html.erb | 77 +++++++++++++++ .../admin/template_promotions/edit.html.erb | 42 ++++++++ .../admin/template_promotions/index.html.erb | 71 ++++++++++++++ .../admin/template_promotions/new.html.erb | 13 +++ config/locales/en.yml | 18 ++++ config/routes.rb | 13 ++- lib/spree/backend/engine.rb | 6 +- ...tion_batch_default_actions_builder_spec.rb | 20 ---- .../default_configuration_builder_spec.rb | 2 +- 23 files changed, 523 insertions(+), 255 deletions(-) create mode 100644 app/controllers/spree/admin/template_promotions_controller.rb delete mode 100644 app/models/spree/admin/actions/promotion_batch_default_actions_builder.rb create mode 100644 app/models/spree/admin/actions/template_promotion_default_actions_builder.rb delete mode 100644 app/views/spree/admin/promotion_batches/edit.html.erb create mode 100644 app/views/spree/admin/promotion_batches/import.html.erb delete mode 100644 app/views/spree/admin/promotion_batches/show.html.erb create mode 100644 app/views/spree/admin/template_promotions/_form.html.erb create mode 100644 app/views/spree/admin/template_promotions/edit.html.erb create mode 100644 app/views/spree/admin/template_promotions/index.html.erb create mode 100644 app/views/spree/admin/template_promotions/new.html.erb delete mode 100644 spec/models/spree/admin/actions/promotion_batch_default_actions_builder_spec.rb diff --git a/app/controllers/spree/admin/promotion_batches_controller.rb b/app/controllers/spree/admin/promotion_batches_controller.rb index d60e3e83b5..6f3ceec1dc 100644 --- a/app/controllers/spree/admin/promotion_batches_controller.rb +++ b/app/controllers/spree/admin/promotion_batches_controller.rb @@ -1,62 +1,46 @@ module Spree module Admin class PromotionBatchesController < ResourceController - def update - if @object.template_promotion_id - flash[:error] = Spree.t(:template_promotion_already_assigned) - respond_with(@object) do |format| - format.html { render action: :edit, status: :unprocessable_entity } - format.js { render layout: false, status: :unprocessable_entity } - end - return - end - super - end - - def destroy - result = Spree::PromotionBatches::Destroy.call(promotion_batch: @promotion_batch) - - if result.success? - flash[:success] = flash_message_for(@promotion_batch, :successfully_removed) - else - flash[:error] = @promotion_batch.errors.full_messages.join(', ') - end - - respond_with(@promotion_batch) do |format| - format.html { redirect_to location_after_destroy } - format.js { render_js_for_destroy } - end - end - - def csv_export - send_data Spree::PromotionBatches::PromotionCodesExporter.new(params).call, - filename: "promo_codes_from_batch_id_#{params[:id]}.csv", - disposition: :attachment, - type: 'text/csv' - end - - def csv_import - file = params[:file] - Spree::PromotionBatches::PromotionCodesImporter.new(file: file, promotion_batch_id: params[:id]).call - redirect_back fallback_location: admin_promotions_path, notice: Spree.t('code_upload') - rescue Spree::PromotionBatches::PromotionCodesImporter::Error => e - redirect_back fallback_location: admin_promotions_path, alert: e.message - end - - def populate - batch_id = params[:id] - options = { - batch_size: params[:batch_size].to_i, - affix: params.dig(:code, :affix)&.to_sym, - content: params[:affix_content], - deny_list: params[:forbidden_phrases].split, - random_part_bytes: params[:random_part_bytes].to_i - } - - Spree::Promotions::PopulatePromotionBatch.new(batch_id, options).call - - flash[:success] = Spree.t('promotion_batch_populated') - redirect_to spree.edit_admin_promotion_batch_url(@promotion_batch) + before_action :set_template_promotion + + def index + @promotion_batches = Spree::PromotionBatch.where(template_promotion: @template_promotion) + end + + def new + @promotion_batch = @template_promotion.promotion_batches.build + end + + def create + Spree::PromotionBatches::CreateWithRandomCodes.new.call(template_promotion: @template_promotion, amount: params[:amount].to_i, random_characters: params[:random_characters].to_i, prefix: params[:prefix], suffix: params[:suffix]) + end + + def import; end + + def process_import + file = params[:file].read + Spree::PromotionBatches::CreateWithCodes.new.call(template_promotion: @template_promotion, codes: file.split("\n")) + redirect_to(admin_template_promotion_promotion_batches_path(template_promotion_id: @template_promotion.id)) + end + + def export + @promotion_batch = @template_promotion.promotion_batches.find(params[:promotion_batch_id]) + csv = Spree::PromotionBatches::Export.new.call(promotion_batch: @promotion_batch) + send_data csv, filename: "codes_#{@promotion_batch}.csv" + end + + private + + def collection_url + admin_template_promotion_promotion_batches_url(@template_promotion) + end + + def new_object_url(options = nil) + new_admin_template_promotion_promotion_batch_url(@template_promotion) + end + + def set_template_promotion + @template_promotion = Spree::Promotion.templates.find(params[:template_promotion_id]) end end end diff --git a/app/controllers/spree/admin/promotions_controller.rb b/app/controllers/spree/admin/promotions_controller.rb index 4b66698654..ac37ce477f 100644 --- a/app/controllers/spree/admin/promotions_controller.rb +++ b/app/controllers/spree/admin/promotions_controller.rb @@ -39,8 +39,7 @@ def collection params[:q] ||= HashWithIndifferentAccess.new params[:q][:s] ||= 'id desc' - @collection = super - @collection = @collection.non_batched + @collection = super.where(template: false) @search = @collection.ransack(params[:q]) @collection = @search.result(distinct: true). includes(promotion_includes). diff --git a/app/controllers/spree/admin/template_promotions_controller.rb b/app/controllers/spree/admin/template_promotions_controller.rb new file mode 100644 index 0000000000..648c18d12d --- /dev/null +++ b/app/controllers/spree/admin/template_promotions_controller.rb @@ -0,0 +1,82 @@ +module Spree + module Admin + class TemplatePromotionsController < ResourceController + before_action :load_data + + def create + invoke_callbacks(:create, :before) + @object.attributes = permitted_resource_params + @object.template = true + if @object.save + invoke_callbacks(:create, :after) + flash[:success] = flash_message_for(@object, :successfully_created) + respond_with(@object) do |format| + format.turbo_stream if create_turbo_stream_enabled? + format.html { redirect_to location_after_save } + format.js { render layout: false } + end + else + invoke_callbacks(:create, :fails) + respond_with(@object) do |format| + format.html { render action: :new, status: :unprocessable_entity } + format.js { render layout: false, status: :unprocessable_entity } + end + end + end + + private + + def new_object_url(options = {}) + spree.new_admin_template_promotion_url(options) + end + + def collection_url(options = {}) + spree.admin_template_promotions_url(options) + end + + def resource + return @resource if @resource + + parent_model_name = parent_data[:model_name] if parent_data + @resource = Spree::Admin::Resource.new 'admin/spree', 'template_promotions', parent_model_name, object_name + end + + def load_data + @actions = Rails.application.config.spree.promotions.actions + + @calculators = Rails.application.config.spree.calculators.promotion_actions_create_adjustments + @promotion_categories = Spree::PromotionCategory.order(:name) + @promotion = @object + end + + def collection + return @collection if defined?(@collection) + + params[:q] ||= HashWithIndifferentAccess.new + params[:q][:s] ||= 'id desc' + + @collection = super + @collection = @collection.templates + @search = @collection.ransack(params[:q]) + @collection = @search.result(distinct: true). + includes(promotion_includes). + page(params[:page]). + per(params[:per_page] || Spree::Backend::Config[:admin_promotions_per_page]) + + @promotions = @collection + end + + def promotion_includes + [:promotion_actions, :promotions_from_template] + end + + def model_class + Spree::Promotion + end + + def permitted_resource_params + params.require(:promotion).permit! + end + end + end +end diff --git a/app/helpers/spree/admin/navigation_helper.rb b/app/helpers/spree/admin/navigation_helper.rb index 7b6c81a7b8..cb698b0fdf 100644 --- a/app/helpers/spree/admin/navigation_helper.rb +++ b/app/helpers/spree/admin/navigation_helper.rb @@ -378,12 +378,12 @@ def product_properties_actions Rails.application.config.spree_backend.actions[:product_properties] end - def promotion_batch_actions - Rails.application.config.spree_backend.actions[:promotion_batch_actions] + def template_promotion_actions + Rails.application.config.spree_backend.actions[:template_promotion] end def promotion_batches_actions - Rails.application.config.spree_backend.actions[:promotion_batches_actions] + Rails.application.config.spree_backend.actions[:promotion_batches] end # rubocop:enable Metrics/ModuleLength end diff --git a/app/models/spree/admin/actions/promotion_batch_default_actions_builder.rb b/app/models/spree/admin/actions/promotion_batch_default_actions_builder.rb deleted file mode 100644 index f9c7979b9d..0000000000 --- a/app/models/spree/admin/actions/promotion_batch_default_actions_builder.rb +++ /dev/null @@ -1,27 +0,0 @@ -module Spree - module Admin - module Actions - class PromotionBatchDefaultActionsBuilder - include Spree::Core::Engine.routes.url_helpers - - def build - root = Root.new - export_codes_to_csv_action(root) - root - end - - private - - def export_codes_to_csv_action(root) - action = - ActionBuilder.new('csv_export', ->(resource) { csv_export_admin_promotion_batch_path(resource) }). - with_icon_key('download.svg'). - with_style(::Spree::Admin::Actions::ActionStyle::PRIMARY). - build - - root.add(action) - end - end - end - end -end diff --git a/app/models/spree/admin/actions/promotion_batches_default_actions_builder.rb b/app/models/spree/admin/actions/promotion_batches_default_actions_builder.rb index 79a6ea955f..03021bb42e 100644 --- a/app/models/spree/admin/actions/promotion_batches_default_actions_builder.rb +++ b/app/models/spree/admin/actions/promotion_batches_default_actions_builder.rb @@ -6,22 +6,49 @@ class PromotionBatchesDefaultActionsBuilder def build root = Root.new - add_new_promotion_batch_action(root) + add_view_promotions_action(root) + add_import_promotion_batch_action(root) + add_generate_promotion_batch_action(root) root end private - def add_new_promotion_batch_action(root) + def add_generate_promotion_batch_action(root) action = - ActionBuilder.new('new_promotion_batch', new_admin_promotion_batch_path). + ActionBuilder.new('generate_codes', ->(template_promotion) { new_admin_template_promotion_promotion_batch_path(template_promotion_id: template_promotion.id) }). + with_label_translation_key('admin.promotion_batches.generate_codes'). with_icon_key('add.svg'). - with_style(::Spree::Admin::Actions::ActionStyle::PRIMARY). - with_create_ability_check(::Spree::PromotionBatch). + with_style(Spree::Admin::Actions::ActionStyle::PRIMARY). + with_create_ability_check(Spree::PromotionBatch). build root.add(action) end + + def add_import_promotion_batch_action(root) + action = + ActionBuilder.new('import_csv', ->(template_promotion) { import_admin_template_promotion_promotion_batches_path(template_promotion_id: template_promotion.id) }). + with_label_translation_key('admin.promotion_batches.import_csv'). + with_icon_key('file-earmark-arrow-up.svg'). + with_style(Spree::Admin::Actions::ActionStyle::LIGHT). + with_create_ability_check(Spree::PromotionBatch). + build + + root.add(action) + end + + def add_view_promotions_action(root) + action = + ActionBuilder.new('view_promotions', ->(template_promotion) { admin_promotions_path(q: { for_template_promotion_id: template_promotion.id }) }). + with_label_translation_key('admin.promotion_batches.view_promotions'). + with_icon_key('list.svg'). + with_style(Spree::Admin::Actions::ActionStyle::LIGHT). + with_manage_ability_check(Spree::Promotion). + build + + root.add(action) + end end end end diff --git a/app/models/spree/admin/actions/template_promotion_default_actions_builder.rb b/app/models/spree/admin/actions/template_promotion_default_actions_builder.rb new file mode 100644 index 0000000000..f452c72310 --- /dev/null +++ b/app/models/spree/admin/actions/template_promotion_default_actions_builder.rb @@ -0,0 +1,55 @@ +module Spree + module Admin + module Actions + class TemplatePromotionDefaultActionsBuilder + include Spree::Core::Engine.routes.url_helpers + + def build + root = Root.new + add_view_promotion_batches_action(root) + add_import_promotion_batch_action(root) + add_generate_promotion_batch_action(root) + root + end + + private + + def add_generate_promotion_batch_action(root) + action = + ActionBuilder.new('generate_codes', ->(template_promotion) { new_admin_template_promotion_promotion_batch_path(template_promotion_id: template_promotion.id) }). + with_label_translation_key('admin.promotion_batches.generate_codes'). + with_icon_key('add.svg'). + with_style(Spree::Admin::Actions::ActionStyle::PRIMARY). + with_create_ability_check(Spree::PromotionBatch). + build + + root.add(action) + end + + def add_import_promotion_batch_action(root) + action = + ActionBuilder.new('import_csv', ->(template_promotion) { import_admin_template_promotion_promotion_batches_path(template_promotion_id: template_promotion.id) }). + with_label_translation_key('admin.promotion_batches.import_csv'). + with_icon_key('file-earmark-arrow-up.svg'). + with_style(Spree::Admin::Actions::ActionStyle::LIGHT). + with_create_ability_check(Spree::PromotionBatch). + build + + root.add(action) + end + + def add_view_promotion_batches_action(root) + action = + ActionBuilder.new('view_promotion_batches', ->(template_promotion) { admin_template_promotion_promotion_batches_path(template_promotion) }). + with_label_translation_key('admin.promotion_batches.view_promotions'). + with_icon_key('list.svg'). + with_style(Spree::Admin::Actions::ActionStyle::LIGHT). + with_manage_ability_check(Spree::Promotion). + build + + root.add(action) + end + end + end + end +end diff --git a/app/models/spree/admin/main_menu/default_configuration_builder.rb b/app/models/spree/admin/main_menu/default_configuration_builder.rb index b524eaada3..5cb2fda22d 100644 --- a/app/models/spree/admin/main_menu/default_configuration_builder.rb +++ b/app/models/spree/admin/main_menu/default_configuration_builder.rb @@ -19,7 +19,6 @@ def build add_integrations_section(root) add_oauth_section(root) add_settings_section(root) - add_promotion_batches_section(root) root end @@ -138,6 +137,10 @@ def add_promotions_section(root) ItemBuilder.new('promotion_categories', admin_promotion_categories_path). with_admin_ability_check(Spree::PromotionCategory). with_label_translation_key('admin.tab.promotion_categories'). + build, + ItemBuilder.new('bulk_promo_codes', admin_template_promotions_path). + with_admin_ability_check(Spree::Promotion, Spree::PromotionBatch). + with_label_translation_key('admin.tab.bulk_promo_codes'). build ] @@ -256,13 +259,6 @@ def add_settings_section(root) build root.add(section) end - - def add_promotion_batches_section(root) - root.add(ItemBuilder.new('promotion_batches', admin_promotion_batches_path). - with_icon_key('stack.svg'). - with_admin_ability_check(Spree::PromotionBatch). - build) - end # rubocop:enable Metrics/AbcSize end # rubocop:enable Metrics/ClassLength diff --git a/app/views/spree/admin/promotion_batches/_form.html.erb b/app/views/spree/admin/promotion_batches/_form.html.erb index c831fea953..279ea2810a 100644 --- a/app/views/spree/admin/promotion_batches/_form.html.erb +++ b/app/views/spree/admin/promotion_batches/_form.html.erb @@ -1,6 +1,26 @@ -
- <%= f.field_container :template_promotion do %> - <%= f.label :template_promotion_id, Spree.t(:template_promotion) %> - <%= f.collection_select :template_promotion_id, Spree::Promotion.non_batched, :id, ->(promotion) { "#{promotion.name} # #{promotion.id}" }, { include_blank: true }, { class: 'select2-clear w-100' } %> - <% end %> +
+
+

<%= Spree.t('admin.promotion_batches.general_settings') %>

+
+ <%= label_tag :amount, Spree.t('admin.promotion_batches.amount') %> + <%= number_field_tag :amount, nil, class: 'form-control' %> +
+ +
+

<%= Spree.t('admin.promotion_batches.generator_settings') %>

+
+ <%= label_tag :random_characters, Spree.t('admin.promotion_batches.random_characters') %> + <%= number_field_tag :random_characters, nil, class: 'form-control' %> +
+ +
+ <%= label_tag :prefix, Spree.t('admin.promotion_batches.prefix') %> + <%= text_field_tag :prefix, nil, class: 'form-control' %> +
+ +
+ <%= label_tag :suffix, Spree.t('admin.promotion_batches.suffix') %> + <%= text_field_tag :suffix, nil, class: 'form-control' %> +
+
diff --git a/app/views/spree/admin/promotion_batches/edit.html.erb b/app/views/spree/admin/promotion_batches/edit.html.erb deleted file mode 100644 index 8c1f21c7f7..0000000000 --- a/app/views/spree/admin/promotion_batches/edit.html.erb +++ /dev/null @@ -1,68 +0,0 @@ -<% content_for :page_title do %> - <%= link_to Spree.t(:promotion_batches), admin_promotion_batches_path %> / - <%= @promotion_batch.id %> -<% end %> - -<% if @promotion_batch.template_promotion_id%> -
-
- <%= form_tag csv_import_admin_promotion_batch_path, method: :post, multipart: true do %> - <%= file_field_tag :file, accept: ".csv" %> - <%= submit_tag Spree.t('promotion_batch_form.import_codes') %> - <% end %> -
-
-<% end %> - -<% unless @promotion_batch.template_promotion_id%> - <%= form_for @promotion_batch, url: object_url, method: :put do |f| %> -
-
- <%= render partial: 'form', locals: { f: f } %> - <%= render partial: 'spree/admin/shared/edit_resource_links' %> -
-
- <% end %> -<% end %> - -<% if @promotion_batch.template_promotion_id %> - <%= form_with url: populate_admin_promotion_batch_path(@promotion_batch), method: :post do %> -
- <%= label_tag :batch_size do %> - <%= number_field_tag(:batch_size) %> - <%= Spree.t('promotion_batch_form.batch_size') %> - <% end %> -
-
- <%= label_tag :prefix do %> - <%= radio_button :code, :affix, 'prefix' %> - <%= Spree.t('promotion_batch_form.prefix') %> - <% end %> -
-
- <%= label_tag :suffix do %> - <%= radio_button :code, :affix, 'suffix' %> - <%= Spree.t('promotion_batch_form.suffix') %> - <% end %> -
-
- <%= label_tag :affix_content do %> - <%= text_field_tag(:affix_content) %> - <%= Spree.t('promotion_batch_form.affix_content') %> - <% end %> -
-
- <%= label_tag :forbidden_phrases do %> - <%= text_area_tag(:forbidden_phrases) %> - <%= Spree.t('promotion_batch_form.forbidden_phrases') %> - <% end %> -
-
- <%= label_tag :random_part_bytes do %> - <%= number_field_tag(:random_part_bytes, value = 4) %> - <%= Spree.t('promotion_batch_form.random_part_bytes') %> - <% end %> -
- <%= submit_tag(Spree.t('promotion_batch_form.populate')) %> - <% end %> -<% end %> diff --git a/app/views/spree/admin/promotion_batches/import.html.erb b/app/views/spree/admin/promotion_batches/import.html.erb new file mode 100644 index 0000000000..39b0c52b47 --- /dev/null +++ b/app/views/spree/admin/promotion_batches/import.html.erb @@ -0,0 +1,24 @@ +<% content_for :page_title do %> + <%= link_to Spree.t(:template_promotions), admin_template_promotions_url %> / + <%= link_to @template_promotion.name, edit_admin_template_promotion_url(@template_promotion) %> / + <%= link_to Spree.t(:promotion_batches), admin_template_promotion_promotion_batches_url(@template_promotion) %> / + <%= Spree.t('admin.promotion_batches.import_csv') %> +<% end %> + + +
+
+ <%= form_tag process_import_admin_template_promotion_promotion_batches_path(@template_promotion), method: :post, multipart: true do %> +
+
+
+ <%= label_tag :file, Spree.t('admin.promotion_batches.csv_file'), class: 'form-control-file' %> + <%= file_field_tag :file, accept: '.csv' %> + <%= Spree.t('admin.promotion_batches.csv_file_notice') %> +
+ <%= render partial: 'spree/admin/shared/new_resource_links' %> +
+
+ <% end %> +
+
diff --git a/app/views/spree/admin/promotion_batches/index.html.erb b/app/views/spree/admin/promotion_batches/index.html.erb index 60b460ae49..a328042c9c 100644 --- a/app/views/spree/admin/promotion_batches/index.html.erb +++ b/app/views/spree/admin/promotion_batches/index.html.erb @@ -1,5 +1,7 @@ <% content_for :page_title do %> - <%= plural_resource_name(Spree::PromotionBatch) %> + <%= link_to Spree.t(:template_promotions), admin_template_promotions_url %> / + <%= link_to @template_promotion.name, edit_admin_template_promotion_url(@template_promotion) %> / + <%= Spree.t(:promotion_batches) %> <% end %> <% content_for :page_actions do %> @@ -7,7 +9,7 @@ <% next unless action.available?(current_ability) %> <%= button_link_to( Spree.t(action.label_translation_key), - action.url, + action.url(@template_promotion), class: action.classes, icon: action.icon_key ) %> @@ -20,26 +22,33 @@ <%= Spree.t(:id) %> - <%= Spree.t(:size) %> - <%= Spree.t(:template_promotion) %> + <%= Spree.t('admin.promotion_batches.generated_codes') %> + <%= Spree.t(:created_at) %> + <%= Spree.t(:state) %> <% @promotion_batches.each do |promotion_batch| %> - <%= link_to Spree::PromotionBatchPresenter.new(promotion_batch).call[:model_name_id], spree.admin_promotion_batch_path(promotion_batch) %> + <%= promotion_batch.id %> <%= promotion_batch.promotions.count %> - <%= link_to Spree::PromotionBatchPresenter.new(promotion_batch).call[:template_promotion_name_id], spree.edit_admin_promotion_path(promotion_batch.template_promotion) if promotion_batch.template_promotion %> - - - <%= link_to_edit promotion_batch, no_text: true if can?(:edit, promotion_batch) %> - <%= link_to_delete promotion_batch, no_text: true if can?(:delete, promotion_batch) %> - + <%= promotion_batch.created_at.strftime('%F %T %Z') %> + <%= promotion_batch.state %> + + <%= button_link_to('', admin_promotions_path({ q: { promotion_batch_id_eq: promotion_batch.id }}), icon: 'list.svg', class: 'btn btn-light btn-sm with-tip icon-link', data: { 'original-title' => Spree.t('admin.promotion_batches.view_promotions') }) %> + <%= button_link_to('', admin_template_promotion_promotion_batch_export_path(template_promotion_id: @template_promotion.id, promotion_batch_id: promotion_batch.id), icon: 'file-earmark-arrow-down.svg', class: 'btn btn-light btn-sm with-tip icon-link', data: { 'original-title' => Spree.t('admin.promotion_batches.export_csv')} ) %> <% end %>
-<% end %> \ No newline at end of file +<% else %> +
+ <%= Spree.t(:no_resource_found, resource: plural_resource_name(Spree::PromotionBatch)) %>, + <%= link_to Spree.t('admin.promotion_batches.generate_codes'), new_object_url if can?(:create, Spree::PromotionBatch) %> + <%= Spree.t(:or) %> + <%= link_to Spree.t('admin.promotion_batches.import_csv'), import_admin_template_promotion_promotion_batches_path(@template_promotion) if can?(:create, Spree::PromotionBatch) %>! +
+<% end %> diff --git a/app/views/spree/admin/promotion_batches/new.html.erb b/app/views/spree/admin/promotion_batches/new.html.erb index 328fe5e0c0..593775e52e 100644 --- a/app/views/spree/admin/promotion_batches/new.html.erb +++ b/app/views/spree/admin/promotion_batches/new.html.erb @@ -1,6 +1,8 @@ <% content_for :page_title do %> - <%= link_to Spree.t(:promotion_batches), admin_promotion_batches_path %> / - <%= Spree.t(:new_promotion_batch) %> + <%= link_to Spree.t(:template_promotions), admin_template_promotions_url %> / + <%= link_to @template_promotion.name, edit_admin_template_promotion_url(@template_promotion) %> / + <%= link_to Spree.t(:promotion_batches), admin_template_promotion_promotion_batches_url(@template_promotion) %> / + <%= Spree.t(:generate_codes) %> <% end %>
diff --git a/app/views/spree/admin/promotion_batches/show.html.erb b/app/views/spree/admin/promotion_batches/show.html.erb deleted file mode 100644 index 2bad5bdcfe..0000000000 --- a/app/views/spree/admin/promotion_batches/show.html.erb +++ /dev/null @@ -1,37 +0,0 @@ -<% content_for :page_title do %> - <%= link_to Spree.t(:promotion_batches), admin_promotion_batches_path %> / - <%= @promotion_batch.id %> -<% end %> - -<% content_for :page_actions do %> - <% promotion_batch_actions.items.each do |action| %> - <% next unless action.available?(current_ability) %> - <%= button_link_to( - Spree.t(action.label_translation_key), - action.url(@promotion_batch), - class: action.classes, - icon: action.icon_key - ) %> - <% end %> -<% end %> - - - - - - - - - - - - <% @promotion_batch.promotions.each do |promotion| %> - - - - - - - <% end %> - -
<%= Spree.t(:code) %><%= Spree.t(:description) %><%= Spree.t(:redeemed) %><%= Spree.t(:expiration) %>
<%= promotion.code %><%= promotion.description %><%= promotion.credits_count == promotion.usage_limit ? Spree.t(:say_yes) : Spree.t(:say_no) %><%= promotion.expires_at.to_date if promotion.expires_at %>
diff --git a/app/views/spree/admin/template_promotions/_form.html.erb b/app/views/spree/admin/template_promotions/_form.html.erb new file mode 100644 index 0000000000..73ea5f4fad --- /dev/null +++ b/app/views/spree/admin/template_promotions/_form.html.erb @@ -0,0 +1,77 @@ +<%= render partial: 'spree/admin/shared/error_messages', locals: { target: @promotion } %> + +
+
+ <%= f.field_container :name do %> + <%= f.label :name %> + <%= f.text_field :name, class: 'form-control' %> + <%= f.error_message_on :name %> + <% end %> + + <%= f.field_container :path do %> + <%= f.label :path %> + <%= f.text_field :path, class: 'form-control' %> + <% end %> + + <%= f.field_container :advertise, class: ['checkbox'] do %> + <%= f.label :advertise do %> + <%= f.check_box :advertise %> + <%= Spree.t(:advertise) %> + <% end %> + <% end %> +
+ +
+ <%= f.field_container :description do %> + <%= f.label :description %> + <%= f.text_area :description, rows: 7, class: 'form-control' %> + <% end %> + + <%= f.field_container :category do %> + <%= f.label :promotion_category %> + <%= f.collection_select(:promotion_category_id, @promotion_categories, :id, :name, { include_blank: Spree.t('match_choices.none') }, { class: 'select2 w-100' }) %> + <% end %> + + <% if @stores.count > 1 %> + <%= f.field_container :stores do %> + <%= f.label :promotion_stores, Spree.t(:stores) %> + <%= collection_select(:promotion, :store_ids, @stores, :id, :unique_name, {}, { multiple: true, class: 'select2' }) %> + <% end %> + <% end %> +
+ +
+
+ <%= f.label :starts_at %> +
+ <%= f.datetime_field :starts_at, + class: 'form-control', + placeholder: Spree.t('starting_from'), + 'data-input':'' %> + + <%= render partial: 'spree/admin/shared/cal_close' %> +
+
+ +
+ <%= f.label :expires_at %> +
+ <%= f.datetime_field :expires_at, + placeholder:Spree.t('ends_at'), + class: 'form-control', + 'data-input':'' %> + + <%= render partial: 'spree/admin/shared/cal_close' %> +
+
+ +
+
diff --git a/app/views/spree/admin/template_promotions/edit.html.erb b/app/views/spree/admin/template_promotions/edit.html.erb new file mode 100644 index 0000000000..2a09dc07bc --- /dev/null +++ b/app/views/spree/admin/template_promotions/edit.html.erb @@ -0,0 +1,42 @@ +<% content_for :page_title do %> + <%= link_to Spree.t(:template_promotions), admin_template_promotions_url %> / + <%= @promotion.name %> +<% end %> + +<% content_for :page_actions do %> + <% template_promotion_actions.items.each do |action| %> + <% next unless action.available?(current_ability) %> + <%= button_link_to( + Spree.t(action.label_translation_key), + action.url(@promotion), + class: action.classes, + icon: action.icon_key + ) %> + <% end %> +<% end %> + +<%= form_for @promotion, url: object_url, method: :put do |f| %> +
+
+ <%= render partial: 'form', locals: { f: f } %> + <%= render partial: 'spree/admin/shared/edit_resource_links' %> +
+
+ +<% end %> + +
+
+
+ <%= render partial: 'spree/admin/promotions/rules' %> +
+
+ +
+
+ <%= render partial: 'spree/admin/promotions/actions' %> +
+
+
+ +<%= render partial: "spree/admin/variants/autocomplete", formats: [:js] %> diff --git a/app/views/spree/admin/template_promotions/index.html.erb b/app/views/spree/admin/template_promotions/index.html.erb new file mode 100644 index 0000000000..1eddb5e7af --- /dev/null +++ b/app/views/spree/admin/template_promotions/index.html.erb @@ -0,0 +1,71 @@ +<% content_for :page_title do %> + <%= Spree.t(:template_promotions) %> +<% end %> + +<% content_for :page_actions do %> + <%= button_link_to Spree.t('admin.template_promotions.new_template'), new_object_url, class: "btn-success", icon: 'add.svg' %> +<% end if can?(:create, Spree::Promotion) %> + +<% content_for :table_filter do %> +
+ <%= search_form_for [:admin, @search] do |f| %> +
+
+
+ <%= label_tag :q_name_cont, Spree.t(:name) %> + <%= f.text_field :name_cont, tabindex: 1, class: "form-control js-quick-search-target js-filterable" %> +
+
+ +
+
+ <%= label_tag :q_promotion_category_id_eq, Spree.t(:promotion_category) %> + <%= f.collection_select(:promotion_category_id_eq, @promotion_categories, :id, :name, { include_blank: Spree.t('match_choices.all') }, { class: 'select2 js-filterable' }) %> +
+
+
+ +
+ <%= button Spree.t(:filter_results), 'search.svg' %> +
+ <% end %> +
+<% end %> + +<% if @promotions.any? %> +
+ + + + + + + + + + + + <% @promotions.each do |template| %> + + + + + + + + <% end %> + +
<%= Spree.t(:name) %><%= Spree.t(:description) %><%= Spree.t('admin.template_promotions.generated_promotions') %><%= Spree.t(:expiration) %>
<%= link_to template.name, spree.edit_admin_template_promotion_path(template) %><%= template.description %><%= link_to template.promotions_from_template.size, admin_template_promotion_promotion_batches_path(template) %><%= template.expires_at.to_date if template.expires_at %> + + <%= link_to_edit template, no_text: true if can?(:edit, template) %> + <%= button_link_to '', admin_template_promotion_promotion_batches_path(template), class: 'btn btn-light btn-sm icon-link with-tip', icon: 'list.svg', data: { 'original-title' => Spree.t('admin.promotion_batches.view_promotions')} %> + +
+
+ <%= render 'spree/admin/shared/index_table_options', collection: @promotions, simple: true %> +<% else %> +
+ <%= Spree.t(:no_resource_found, resource: plural_resource_name(Spree::Promotion)) %>, + <%= link_to Spree.t(:add_one), new_object_url if can?(:create, Spree::Promotion) %>! +
+<% end %> diff --git a/app/views/spree/admin/template_promotions/new.html.erb b/app/views/spree/admin/template_promotions/new.html.erb new file mode 100644 index 0000000000..58086c09e8 --- /dev/null +++ b/app/views/spree/admin/template_promotions/new.html.erb @@ -0,0 +1,13 @@ +<% content_for :page_title do %> + <%= link_to Spree.t(:template_promotions), admin_template_promotions_url %> / + <%= Spree.t('admin.template_promotions.new_template') %> +<% end %> + +
+
+ <%= form_for :promotion, url: collection_url do |f| %> + <%= render partial: 'form', locals: { f: f } %> + <%= render partial: 'spree/admin/shared/new_resource_links' %> + <% end %> +
+
diff --git a/config/locales/en.yml b/config/locales/en.yml index ff9a4fb6a0..f283efbd0c 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -116,6 +116,7 @@ en: physical: Physical digital: Digital tab: + bulk_promo_codes: Bulk Promo Codes configuration: Configuration content: Content option_types: Option Types @@ -171,6 +172,20 @@ en: scopes: Scopes documentation_message: 'To learn how to authenticate requests to the Platform API, ' documentation_cta: see the documentation + promotion_batches: + amount: Number of codes to generate + csv_file: Select a CSV file to import + csv_file_notice: The CSV file should include one promo code per line, without any additional columns + export_csv: Export CSV + generate_codes: Generate Codes + generated_codes: Generated Codes + general_settings: General Settings + generator_settings: Generator Settings + import_csv: Import CSV + prefix: (Optional) Prefix + random_characters: Number of random characters to generate for each code + suffix: (Optional) Suffix + view_promotions: View Promotions reports: for: For %{store_name} return_authorization: @@ -190,6 +205,9 @@ en: customer_support_email_help: "This email is visible to your Store visitors in the Footer section" new_order_notifications_email_help: "If you want to receive an email notification every time someone places an Order please provide an email address for that notification to be sent to" seo_robots: "Please check this page for more help" + template_promotions: + generated_promotions: Generated Promotions + new_template: New Template user: account: Account addresses: Addresses diff --git a/config/routes.rb b/config/routes.rb index c1ee28aa19..a469e83403 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -10,11 +10,14 @@ resources :promotion_categories, except: [:show] - resources :promotion_batches do - member do - get :csv_export, to: 'promotion_batches#csv_export' - post :csv_import, to: 'promotion_batches#csv_import' - post :populate + resources :template_promotions do + resources :promotion_batches, only: %i[index new create show] do + collection do + get :import + post :process_import, to: 'promotion_batches#process_import' + end + + get :export end end diff --git a/lib/spree/backend/engine.rb b/lib/spree/backend/engine.rb index 119358659a..73eb404447 100644 --- a/lib/spree/backend/engine.rb +++ b/lib/spree/backend/engine.rb @@ -48,10 +48,8 @@ class Engine < ::Rails::Engine Rails.application.config.spree_backend.actions[:payments] = Spree::Admin::Actions::PaymentsDefaultActionsBuilder.new.build Rails.application.config.spree_backend.actions[:variants] = Spree::Admin::Actions::VariantsDefaultActionsBuilder.new.build Rails.application.config.spree_backend.actions[:product_properties] = Spree::Admin::Actions::ProductPropertiesDefaultActionsBuilder.new.build - Rails.application.config.spree_backend.actions[:promotion_batch_actions] = - Spree::Admin::Actions::PromotionBatchDefaultActionsBuilder.new.build - Rails.application.config.spree_backend.actions[:promotion_batches_actions] = - Spree::Admin::Actions::PromotionBatchesDefaultActionsBuilder.new.build + Rails.application.config.spree_backend.actions[:template_promotion] = Spree::Admin::Actions::TemplatePromotionDefaultActionsBuilder.new.build + Rails.application.config.spree_backend.actions[:promotion_batches] = Spree::Admin::Actions::PromotionBatchesDefaultActionsBuilder.new.build end end end diff --git a/spec/models/spree/admin/actions/promotion_batch_default_actions_builder_spec.rb b/spec/models/spree/admin/actions/promotion_batch_default_actions_builder_spec.rb deleted file mode 100644 index 2f45ca3460..0000000000 --- a/spec/models/spree/admin/actions/promotion_batch_default_actions_builder_spec.rb +++ /dev/null @@ -1,20 +0,0 @@ -require 'spec_helper' - -module Spree - module Admin - describe Actions::PromotionBatchDefaultActionsBuilder, type: :model do - let(:builder) { described_class.new } - let(:default_actions) do - %w(csv_export) - end - - describe '#build' do - subject { builder.build } - - it 'builds default tabs' do - expect(subject.items.map(&:key)).to match(default_actions) - end - end - end - end -end diff --git a/spec/models/spree/admin/main_menu/default_configuration_builder_spec.rb b/spec/models/spree/admin/main_menu/default_configuration_builder_spec.rb index 455a3fec23..353f05a9dc 100644 --- a/spec/models/spree/admin/main_menu/default_configuration_builder_spec.rb +++ b/spec/models/spree/admin/main_menu/default_configuration_builder_spec.rb @@ -9,7 +9,7 @@ module Admin subject { builder.build } it 'builds a valid menu' do - expect(subject.items.count).to eq(13) + expect(subject.items.count).to eq(12) expect(subject.items.map(&:key)).to include('dashboard') expect(subject.items.map(&:key)).to include('orders') expect(subject.items.map(&:key)).to include('settings') From 22880e766fc676022bcb1e53c2bd519aa3e7f680 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mateusz=20Por=C4=99bski?= Date: Tue, 5 Mar 2024 09:00:50 +0100 Subject: [PATCH 3/4] cosmetic touch-ups - fix missing translations, missing required fields properties and routes errors --- .../spree/admin/promotion_batches_controller.rb | 1 + .../actions/template_promotion_default_actions_builder.rb | 2 +- app/views/spree/admin/promotion_batches/_form.html.erb | 8 ++++---- app/views/spree/admin/promotion_batches/import.html.erb | 4 ++-- app/views/spree/admin/promotion_batches/index.html.erb | 2 +- app/views/spree/admin/promotion_batches/new.html.erb | 2 +- app/views/spree/admin/promotions/_actions.html.erb | 2 +- app/views/spree/admin/promotions/_rules.html.erb | 2 +- config/locales/en.yml | 2 ++ 9 files changed, 14 insertions(+), 11 deletions(-) diff --git a/app/controllers/spree/admin/promotion_batches_controller.rb b/app/controllers/spree/admin/promotion_batches_controller.rb index 6f3ceec1dc..7a36955bdf 100644 --- a/app/controllers/spree/admin/promotion_batches_controller.rb +++ b/app/controllers/spree/admin/promotion_batches_controller.rb @@ -13,6 +13,7 @@ def new def create Spree::PromotionBatches::CreateWithRandomCodes.new.call(template_promotion: @template_promotion, amount: params[:amount].to_i, random_characters: params[:random_characters].to_i, prefix: params[:prefix], suffix: params[:suffix]) + redirect_to(admin_template_promotion_promotion_batches_path(template_promotion_id: @template_promotion.id)) end def import; end diff --git a/app/models/spree/admin/actions/template_promotion_default_actions_builder.rb b/app/models/spree/admin/actions/template_promotion_default_actions_builder.rb index f452c72310..c8fc88d231 100644 --- a/app/models/spree/admin/actions/template_promotion_default_actions_builder.rb +++ b/app/models/spree/admin/actions/template_promotion_default_actions_builder.rb @@ -41,7 +41,7 @@ def add_import_promotion_batch_action(root) def add_view_promotion_batches_action(root) action = ActionBuilder.new('view_promotion_batches', ->(template_promotion) { admin_template_promotion_promotion_batches_path(template_promotion) }). - with_label_translation_key('admin.promotion_batches.view_promotions'). + with_label_translation_key('admin.promotion_batches.view_promotion_batches'). with_icon_key('list.svg'). with_style(Spree::Admin::Actions::ActionStyle::LIGHT). with_manage_ability_check(Spree::Promotion). diff --git a/app/views/spree/admin/promotion_batches/_form.html.erb b/app/views/spree/admin/promotion_batches/_form.html.erb index 279ea2810a..d561f76711 100644 --- a/app/views/spree/admin/promotion_batches/_form.html.erb +++ b/app/views/spree/admin/promotion_batches/_form.html.erb @@ -2,15 +2,15 @@

<%= Spree.t('admin.promotion_batches.general_settings') %>

- <%= label_tag :amount, Spree.t('admin.promotion_batches.amount') %> - <%= number_field_tag :amount, nil, class: 'form-control' %> + <%= label_tag :amount, raw(Spree.t('admin.promotion_batches.amount') + required_span_tag) %> + <%= number_field_tag :amount, nil, class: 'form-control', required: :required %>

<%= Spree.t('admin.promotion_batches.generator_settings') %>

- <%= label_tag :random_characters, Spree.t('admin.promotion_batches.random_characters') %> - <%= number_field_tag :random_characters, nil, class: 'form-control' %> + <%= label_tag :random_characters, raw(Spree.t('admin.promotion_batches.random_characters') + required_span_tag) %> + <%= number_field_tag :random_characters, nil, class: 'form-control', required: :required %>
diff --git a/app/views/spree/admin/promotion_batches/import.html.erb b/app/views/spree/admin/promotion_batches/import.html.erb index 39b0c52b47..17ca8118eb 100644 --- a/app/views/spree/admin/promotion_batches/import.html.erb +++ b/app/views/spree/admin/promotion_batches/import.html.erb @@ -12,8 +12,8 @@
- <%= label_tag :file, Spree.t('admin.promotion_batches.csv_file'), class: 'form-control-file' %> - <%= file_field_tag :file, accept: '.csv' %> + <%= label_tag :file, raw(Spree.t('admin.promotion_batches.csv_file') + required_span_tag), class: 'form-control-file' %> + <%= file_field_tag :file, accept: '.csv', required: :required %> <%= Spree.t('admin.promotion_batches.csv_file_notice') %>
<%= render partial: 'spree/admin/shared/new_resource_links' %> diff --git a/app/views/spree/admin/promotion_batches/index.html.erb b/app/views/spree/admin/promotion_batches/index.html.erb index a328042c9c..208d0d4601 100644 --- a/app/views/spree/admin/promotion_batches/index.html.erb +++ b/app/views/spree/admin/promotion_batches/index.html.erb @@ -21,7 +21,7 @@ - + diff --git a/app/views/spree/admin/promotion_batches/new.html.erb b/app/views/spree/admin/promotion_batches/new.html.erb index 593775e52e..5f7f0b63d8 100644 --- a/app/views/spree/admin/promotion_batches/new.html.erb +++ b/app/views/spree/admin/promotion_batches/new.html.erb @@ -2,7 +2,7 @@ <%= link_to Spree.t(:template_promotions), admin_template_promotions_url %> / <%= link_to @template_promotion.name, edit_admin_template_promotion_url(@template_promotion) %> / <%= link_to Spree.t(:promotion_batches), admin_template_promotion_promotion_batches_url(@template_promotion) %> / - <%= Spree.t(:generate_codes) %> + <%= Spree.t('admin.promotion_batches.generate_codes') %> <% end %>
diff --git a/app/views/spree/admin/promotions/_actions.html.erb b/app/views/spree/admin/promotions/_actions.html.erb index 0e2e5a3566..3a05401a9c 100644 --- a/app/views/spree/admin/promotions/_actions.html.erb +++ b/app/views/spree/admin/promotions/_actions.html.erb @@ -20,7 +20,7 @@ <%= form_for @promotion, url: spree.admin_promotion_path(@promotion), method: :put do |f| %>
<% if @promotion.actions.any? %> - <%= render partial: 'promotion_action', collection: @promotion.actions %> + <%= render partial: 'spree/admin/promotions/promotion_action', collection: @promotion.actions %> <% else %>
<%= Spree.t(:no_actions_added) %> diff --git a/app/views/spree/admin/promotions/_rules.html.erb b/app/views/spree/admin/promotions/_rules.html.erb index ccf4209a18..bc43632da3 100644 --- a/app/views/spree/admin/promotions/_rules.html.erb +++ b/app/views/spree/admin/promotions/_rules.html.erb @@ -31,7 +31,7 @@
<% if @promotion.rules.any? %> - <%= render partial: 'promotion_rule', collection: @promotion.rules, locals: {} %> + <%= render partial: 'spree/admin/promotions/promotion_rule', collection: @promotion.rules, locals: {} %> <% else %>
<%= Spree.t(:no_rules_added) %> diff --git a/config/locales/en.yml b/config/locales/en.yml index f283efbd0c..381a17bdc2 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -1,6 +1,7 @@ en: spree: admin: + id: ID url: URL active: Active name: Name @@ -186,6 +187,7 @@ en: random_characters: Number of random characters to generate for each code suffix: (Optional) Suffix view_promotions: View Promotions + view_promotion_batches: View Promotion Batches reports: for: For %{store_name} return_authorization: From 46eca221eb7b9fa8dec2aa6f4878b840c0db6fbc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mateusz=20Por=C4=99bski?= Date: Tue, 5 Mar 2024 09:37:41 +0100 Subject: [PATCH 4/4] tests for template promotions and promotion batches features --- .../promotion_batches_spec.rb | 145 ++++++++++++++++++ .../template_promotions_spec.rb | 74 +++++++++ spec/support/promotion_batch_codes.csv | 2 + 3 files changed, 221 insertions(+) create mode 100644 spec/features/admin/bulk_promo_codes/promotion_batches_spec.rb create mode 100644 spec/features/admin/bulk_promo_codes/template_promotions_spec.rb create mode 100644 spec/support/promotion_batch_codes.csv diff --git a/spec/features/admin/bulk_promo_codes/promotion_batches_spec.rb b/spec/features/admin/bulk_promo_codes/promotion_batches_spec.rb new file mode 100644 index 0000000000..25ad13a1f7 --- /dev/null +++ b/spec/features/admin/bulk_promo_codes/promotion_batches_spec.rb @@ -0,0 +1,145 @@ +require 'spec_helper' + +describe 'New Promotion Batches', type: :feature, js: true do + stub_authorization! + + context 'when object is created' do + let!(:template_promotion) { create(:promotion, name: 'Template Promotion', template: true) } + + before do + visit spree.new_admin_template_promotion_promotion_batch_path(template_promotion_id: template_promotion.id) + fill_in 'Number of codes to generate', with: 2 + fill_in 'Number of random characters to generate for each code', with: 10 + end + + it 'have specified number of codes' do + click_button 'Create' + wait_for_turbo + + expect(template_promotion.promotion_batches.length).to eq(1) + + promotion_batch = template_promotion.promotion_batches.first + + expect(promotion_batch.codes.length).to eq(2) + end + + it 'have codes with specified length' do + click_button 'Create' + wait_for_turbo + + expect(template_promotion.promotion_batches.length).to eq(1) + promotion_batch = template_promotion.promotion_batches.first + promotion_batch.codes.each do |code| + expect(code.length).to eq(10) + end + end + + it 'have specified suffixes and prefixes' do + prefix, suffix = 'prefix', 'suffix' + + fill_in '(Optional) Prefix', with: prefix + fill_in '(Optional) Suffix', with: suffix + + click_button 'Create' + wait_for_turbo + + promotion_batch = template_promotion.promotion_batches.first + + promotion_batch.codes.each do |code| + expect(code).to match("^#{prefix}[A-Z0-9]{10}#{suffix}$") + end + end + end + + context 'create promotions' do + let!(:template_promotion) { + create(:promotion, + name: 'Template Promotion', + description: 'Template Promotion Description', + template: true, + starts_at: DateTime.new(2012, 1, 18, 16, 45), + expires_at: DateTime.new(2012, 1, 18, 16, 45)) + } + + before do + visit spree.new_admin_template_promotion_promotion_batch_path(template_promotion_id: template_promotion.id) + fill_in 'Number of codes to generate', with: 2 + fill_in 'Number of random characters to generate for each code', with: 10 + end + + it 'in specified number' do + click_button 'Create' + wait_for_turbo + + promotion_batch = template_promotion.promotion_batches.first + expect(promotion_batch.state).to eq('completed') + expect(promotion_batch.promotions.count).to eq(2) + end + + it 'that have the same properties as the template promotion' do + click_button 'Create' + wait_for_turbo + + promotion_batch = template_promotion.promotion_batches.first + expect(promotion_batch.state).to eq('completed') + + promotion_batch.promotions.each do |promotion| + expect(promotion.name).to eq(template_promotion.name) + expect(promotion.description).to eq(template_promotion.description) + expect(promotion.starts_at).to eq(template_promotion.starts_at) + expect(promotion.expires_at).to eq(template_promotion.expires_at) + end + end + end + + context 'after creating promotions' do + + let!(:template_promotion1) { create(:promotion, name: 'Template Promotion 1', template: true) } + let!(:template_promotion2) { create(:promotion, name: 'Template Promotion 2', template: true) } + + before do + [template_promotion1, template_promotion2].each do |template| + 2.times do + visit spree.new_admin_template_promotion_promotion_batch_path(template_promotion_id: template.id) + fill_in 'Number of codes to generate', with: 2 + fill_in 'Number of random characters to generate for each code', with: 10 + click_button 'Create' + wait_for_turbo + end + end + + end + it 'allows displaying created promotions filtered by template promotion' do + visit spree.admin_promotions_path({ q: { for_template_promotion_id: template_promotion1.id } }) + expect(page).to have_content('Template Promotion 1', minimum: 4) + expect(page).to have_no_content('Template Promotion 2') + end + + it 'allows displaying created promotions filtered by promotion batch' do + visit spree.admin_promotions_path({ q: { promotion_batch_id_eq: template_promotion1.promotion_batches.first.id } }) + expect(page).to have_content('Template Promotion 1', maximum: 2) + expect(page).to have_no_content('Template Promotion 2') + end + + end + + context 'CSV files' do + let!(:template_promotion) { create(:promotion, name: 'Template Promotion', template: true) } + + let(:file_path) { Rails.root + '../../spec/support/promotion_batch_codes.csv' } + + before do + visit spree.import_admin_template_promotion_promotion_batches_path(template_promotion_id: template_promotion.id) + end + + it 'import properly' do + attach_file('file', file_path) + click_button 'Create' + wait_for_turbo + + promotion_batch = Spree::PromotionBatch.first + expect(promotion_batch.codes).to match_array(['CODE1', 'CODE2']) + end + end +end + diff --git a/spec/features/admin/bulk_promo_codes/template_promotions_spec.rb b/spec/features/admin/bulk_promo_codes/template_promotions_spec.rb new file mode 100644 index 0000000000..13ce5b2211 --- /dev/null +++ b/spec/features/admin/bulk_promo_codes/template_promotions_spec.rb @@ -0,0 +1,74 @@ +require 'spec_helper' + +describe 'Template Promotion', type: :feature, js: true do + stub_authorization! + + let(:store) { Spree::Store.default } + let!(:store_2) { create(:store) } + let(:promotion_name) { 'Template Promotion' } + + context 'is distinct from regular promotions' do + before do + visit spree.new_admin_template_promotion_path + end + + it 'has template property equal to true' do + fill_in 'Name', with: promotion_name + + click_button 'Create' + wait_for_turbo + + promotion = store.promotions.find_by!(name: promotion_name) + expect(promotion.template).to eq(true) + end + + it "doesn't allow specifying a code" do + expect(page).to have_no_content('Code') + end + + it "doesn't appear in promotions list" do + fill_in 'Name', with: promotion_name + + click_button 'Create' + wait_for_turbo + + promotion = store.promotions.find_by!(name: promotion_name) + expect(promotion.name).to eq(promotion_name) + visit spree.admin_promotions_path + expect(page).to have_no_content(promotion_name) + end + end + + context 'while being created' do + before do + visit spree.new_admin_template_promotion_path + end + + it 'allows you to set a promotion with start and end time' do + fill_in 'Name', with: promotion_name + + fill_in_date_picker('promotion_starts_at', { year: 2012, month: 1, day: 18, hour: 16, minute: 45 }) + fill_in_date_picker('promotion_expires_at', { year: 2013, month: 3, day: 25, hour: 22, minute: 10 }) + + click_button 'Create' + wait_for_turbo + + promotion = store.promotions.find_by!(name: promotion_name) + expect(promotion.starts_at).to eq(DateTime.new(2012, 1, 18, 16, 45)) + expect(promotion.expires_at).to eq(DateTime.new(2013, 3, 25, 22, 10)) + end + + it 'allows assigning multiple stores' do + fill_in 'Name', with: promotion_name + select2 store_2.unique_name, from: 'Stores' + + click_button 'Create' + wait_for_turbo + + expect(page).to have_content('successfully created') + + promotion = store.promotions.find_by!(name: promotion_name) + expect(promotion.stores).to contain_exactly(store, store_2) + end + end +end diff --git a/spec/support/promotion_batch_codes.csv b/spec/support/promotion_batch_codes.csv new file mode 100644 index 0000000000..651e5a4dc8 --- /dev/null +++ b/spec/support/promotion_batch_codes.csv @@ -0,0 +1,2 @@ +CODE1 +CODE2
<%= Spree.t(:id) %><%= Spree.t('admin.id') %> <%= Spree.t('admin.promotion_batches.generated_codes') %> <%= Spree.t(:created_at) %> <%= Spree.t(:state) %>