0
0
Fork 1
Spiegel von https://github.com/paviliondev/discourse-custom-wizard.git synchronisiert 2025-01-22 15:59:00 +01:00
discourse-custom-wizard/lib/custom_wizard/builder.rb
2024-10-16 13:52:03 +02:00

342 Zeilen
9,2 KiB
Ruby

# frozen_string_literal: true
class CustomWizard::Builder
attr_accessor :wizard, :updater, :template
def initialize(wizard_id, user = nil, guest_id = nil)
@template = CustomWizard::Template.create(wizard_id)
return nil if @template.nil?
@wizard = CustomWizard::Wizard.new(template.data, user, guest_id)
end
def self.sorted_handlers
@sorted_handlers ||= []
end
def self.step_handlers
sorted_handlers.map { |h| { wizard_id: h[:wizard_id], block: h[:block] } }
end
def self.add_step_handler(priority = 0, wizard_id, &block)
sorted_handlers << { priority: priority, wizard_id: wizard_id, block: block }
@sorted_handlers.sort_by! { |h| -h[:priority] }
end
def build(build_opts = {}, params = {})
return nil if !SiteSetting.custom_wizard_enabled || !@wizard
return @wizard if !@wizard.can_access? && !build_opts[:force]
build_opts[:reset] = build_opts[:reset] || @wizard.restart_on_revisit
@template.steps.each do |step_template|
next if !check_condition(step_template)
@wizard.append_step(step_template["id"]) do |step|
step = check_if_permitted(step, step_template)
next if !step.permitted
save_permitted_params(step_template, params)
step = add_step_attributes(step, step_template)
step = append_step_fields(step, step_template, build_opts)
step.on_update do |updater|
@updater = updater
@submission = @wizard.current_submission
@submission.fields.merge!(@updater.submission)
@updater.validate
next if @updater.errors.any?
apply_step_handlers
next if @updater.errors.any?
run_step_actions
if @updater.errors.empty?
route_to = @submission.route_to
@submission.route_to = nil
@submission.save
@wizard.update!
@updater.result[:redirect_on_next] = route_to if route_to
true
else
false
end
end
end
end
@wizard.update!
CustomWizard::Submission.cleanup_incomplete_submissions(@wizard)
@wizard
end
def check_condition(template)
if template["condition"].present?
result =
CustomWizard::Mapper.new(
inputs: template["condition"],
user: @wizard.user,
data: @wizard.current_submission&.fields_and_meta,
opts: {
multiple: true,
},
).perform
result.any?
else
true
end
end
private
def mapper
CustomWizard::Mapper.new(user: @wizard.user, data: @wizard.current_submission&.fields_and_meta)
end
def append_field(step, step_template, field_template, build_opts)
params = {
id: field_template["id"],
type: field_template["type"],
required: field_template["required"],
}
%w[
label
description
image
key
validations
min_length
max_length
char_counter
tag_groups
].each { |key| params[key.to_sym] = field_template[key] if field_template[key] }
params[:value] = prefill_field(field_template, step_template)
if !build_opts[:reset] && (submission = @wizard.current_submission).present?
params[:value] = submission.fields[field_template["id"]] if submission.fields[
field_template["id"]
]
end
if field_template["type"] === "group" && params[:value].present?
params[:value] = params[:value].first
end
params[:value] = standardise_boolean(params[:value]) if field_template["type"] === "checkbox"
params[:file_types] = field_template["file_types"] if field_template["type"] === "upload"
if %w[date time date_time].include?(field_template["type"])
params[:format] = field_template["format"]
end
if %w[category tag topic].include?(field_template["type"])
params[:limit] = field_template["limit"]
end
if field_template["type"] === "tag"
params[:can_create_tag] = standardise_boolean(field_template["can_create_tag"])
end
params[:property] = field_template["property"] if field_template["type"] === "category"
params[:category] = field_template["category"] if field_template["type"] === "topic"
if (content_inputs = field_template["content"]).present?
content =
CustomWizard::Mapper.new(
inputs: content_inputs,
user: @wizard.user,
data: @wizard.current_submission&.fields_and_meta,
opts: {
with_type: true,
},
).perform
if content.present? && content[:result].present?
if content[:type] == "association"
content[:result] = content[:result].map { |item| { id: item[:key], name: item[:value] } }
end
params[:content] = content[:result]
end
end
if field_template["index"].present?
index =
CustomWizard::Mapper.new(
inputs: field_template["index"],
user: @wizard.user,
data: @wizard.current_submission&.fields_and_meta,
).perform
params[:index] = index.to_i unless index.nil?
end
if field_template["description"].present?
params[:description] = mapper.interpolate(
field_template["description"],
user: @wizard.user,
value: true,
wizard: true,
template: true,
)
end
if field_template["preview_template"].present?
preview_template =
mapper.interpolate(
field_template["preview_template"],
user: @wizard.user,
value: true,
wizard: true,
template: true,
)
params[:preview_template] = PrettyText.cook(preview_template)
end
if field_template["placeholder"].present?
params[:placeholder] = mapper.interpolate(
field_template["placeholder"],
user: @wizard.user,
value: true,
wizard: true,
template: true,
)
end
field = step.add_field(params)
end
def prefill_field(field_template, step_template)
if (prefill = field_template["prefill"]).present?
CustomWizard::Mapper.new(
inputs: prefill,
user: @wizard.user,
data: @wizard.current_submission&.fields_and_meta,
).perform
end
end
def check_if_permitted(step, step_template)
step.permitted = true
step = ensure_required_data(step, step_template) if step_template["required_data"]
if !step.permitted
if step_template["required_data_message"]
step.permitted_message = step_template["required_data_message"]
end
end
step
end
def add_step_attributes(step, step_template)
%w[index title banner key force_final].each do |attr|
step.send("#{attr}=", step_template[attr]) if step_template[attr]
end
if step_template["description"]
step.description =
mapper.interpolate(
step_template["description"],
user: @wizard.user,
value: true,
wizard: true,
template: true,
)
step.description = PrettyText.cook(step.description)
end
step
end
def append_step_fields(step, step_template, build_opts)
if step_template["fields"] && step_template["fields"].length
step_template["fields"].each do |field_template|
next if !check_condition(field_template)
append_field(step, step_template, field_template, build_opts)
end
end
step.update_field_order!
step
end
def standardise_boolean(value)
ActiveRecord::Type::Boolean.new.cast(value)
end
def save_permitted_params(step_template, params)
return if step_template["permitted_params"].blank?
permitted_params = step_template["permitted_params"]
permitted_data = {}
submission_key = nil
params_key = nil
submission = @wizard.current_submission
permitted_params.each do |pp|
pair = pp["pairs"].first
params_key = pair["key"].to_sym
submission_key = pair["value"].to_sym
if submission_key && params_key && params[params_key].present?
submission.permitted_param_keys << submission_key.to_s
submission.fields[submission_key] = params[params_key]
end
end
submission.save
end
def ensure_required_data(step, step_template)
step_template["required_data"].each do |required|
pairs = required["pairs"].select { |pair| pair["key"].present? && pair["value"].present? }
if pairs.any? && !@wizard.current_submission.present?
step.permitted = false
break
end
pairs.each { |pair| pair["key"] = @wizard.current_submission.fields[pair["key"]] }
if !mapper.validate_pairs(pairs)
step.permitted = false
break
end
end
step
end
def apply_step_handlers
CustomWizard::Builder.step_handlers.each do |handler|
handler[:block].call(self) if handler[:wizard_id] == @wizard.id
end
end
def run_step_actions
if @template.actions.present?
@template.actions.each do |action_template|
if action_template["run_after"] === updater.step.id
result =
CustomWizard::Action.new(
action: action_template,
wizard: @wizard,
submission: @submission,
).perform
@submission = result.submission if result.success?
end
end
end
end
end