module RocketJobMissionControl module ApplicationHelper STATE_ICON_MAP = { aborted: "fas fa-stop", completed: "fas fa-check", disabled: "fas fa-stop", enabled: "fas fa-check", failed: "fas fa-exclamation-triangle", paused: "fas fa-pause", pending: "fas fa-inbox", queued: "fas fa-inbox", running: "fas fa-play", sleeping: "fas fa-hourglass", scheduled: "fas fa-clock", starting: "fas fa-cogs", stopping: "fas fa-stop", zombie: "fas fa-hourglass" }.freeze def state_icon(state) STATE_ICON_MAP[state.to_sym] + " " + state.to_s end def site_title "Rocket Job Mission Control" end def title @page_title ||= params[:controller].to_s.titleize h(@full_title || [@page_title, site_title].compact.join(" | ")) end def active_page(path) "active" if current_page?(path) end def pretty_print_array_or_hash(arguments) return arguments unless arguments.is_a?(Array) || arguments.is_a?(Hash) json_string_options = {space: " ", indent: " ", array_nl: "
", object_nl: "
"} JSON.generate(arguments, json_string_options).html_safe end # Returns [Array] list of inclusion values for this attribute. # Returns nil when there are no inclusion values for this attribute. def extract_inclusion_values(klass, attribute) values = nil klass.validators_on(attribute).each do |validator| case validator when ActiveModel::Validations::InclusionValidator values = validator.options[:in] end end values end # Returns the editable field as html for use in editing dynamic fields from a Job class. def editable_field_html(klass, field_name, value, f) # When editing a job the values are of the correct type. # When editing a dirmon entry values are strings. field = klass.fields[field_name.to_s] return unless field&.type placeholder = field.default_val placeholder = nil if placeholder.is_a?(Proc) case field.type.name when "Integer" options = extract_inclusion_values(klass, field_name) f.number_field(field_name, in: options, include_blank: false, value: value, class: "form-control", placeholder: placeholder) when "String", "Symbol", "Mongoid::StringifiedSymbol" options = extract_inclusion_values(klass, field_name) if options f.select(field_name, options, {include_blank: options.include?(nil), selected: value}, {class: "selectize form-control"}) else f.text_field(field_name, value: value, class: "form-control", placeholder: placeholder) end when "Boolean", "Mongoid::Boolean" options = extract_inclusion_values(klass, field_name) || [nil, "true", "false"] f.select(field_name, options, {include_blank: options.include?(nil), selected: value}, {class: "selectize form-control"}) when "Hash" "[JSON Hash]\n".html_safe + f.text_field(field_name, value: value ? value.to_json : "", class: "form-control", placeholder: '{"key1":"value1", "key2":"value2", "key3":"value3"}') when "Array" options = Array(value) f.select(field_name, options_for_select(options, options), {include_hidden: false}, {class: "selectize form-control", multiple: true}) else "[#{field.type.name}]".html_safe + f.text_field(field_name, value: value, class: "form-control", placeholder: placeholder) end end # This method creates a link with `data-id` `data-fields` attributes. These attributes are used to create new instances of the nested fields through Javascript. def link_to_add_fields(name, f, association, option) # Takes an object (@job) and creates a new instance of its associated model (:properties) new_object = f.object.send(association).klass.new # Saves the unique ID of the object into a variable. # This is needed to ensure the key of the associated array is unique. This is makes parsing the content in the `data-fields` attribute easier through Javascript. # We could use another method to achive this. id = new_object.object_id # https://api.rubyonrails.org/ fields_for(record_name, record_object = nil, fields_options = {}, &block) # record_name = :addresses # record_object = new_object # fields_options = { child_index: id } # child_index` is used to ensure the key of the associated array is unique, and that it matched the value in the `data-id` attribute. # `person[addresses_attributes][child_index_value][_destroy]` fields = f.fields_for(association, new_object, child_index: id) do |builder| # `association.to_s.singularize + "_fields"` ends up evaluating to `address_fields` # The render function will then look for `views/people/_address_fields.html.erb` # The render function also needs to be passed the value of 'builder', because `views/dirmon_entries/_input_categories.html.erb` needs this to render the form tags. render(association.to_s.singularize + "_fields", f: builder) end # This renders a simple link, but passes information into `data` attributes. # This info can be named anything we want, but in this case we chose `data-id:` and `data-fields:`. # The `id:` is from `new_object.object_id`. # The `fields:` are rendered from the `fields` blocks. # We use `gsub("\n", "")` to remove anywhite space from the rendered partial. # The `id:` value needs to match the value used in `child_index: id`. link_to(name, '#', class: "add_fields btn btn-#{option}", data: { id: id, fields: fields.gsub("\n", "") }) end end end