app/models/effective/form_inputs/collection_input.rb



# frozen_string_literal: true

module Effective
  module FormInputs
    class CollectionInput < Effective::FormInput
      def initialize(name, options, builder:, html_options: nil)
        super
        assign_options_collection!
        assign_options_collection_methods!
      end

      def polymorphic?
        return @polymorphic unless @polymorphic.nil?
        @polymorphic = (options.delete(:polymorphic) || false)
      end

      def grouped?
        return @grouped unless @grouped.nil?
        @grouped = (options.delete(:grouped) || false)
      end

      def custom? # default true
        return @custom unless @custom.nil?
        @custom = (options.delete(:custom) != false)
      end

      def inline? # default false
        return @inline unless @inline.nil?
        @inline = (options[:input].delete(:inline) == true)
      end

      def collection_options
        return @collection_options unless @collection_options.nil?

        checked = options[:input].delete(:checked)
        selected = options[:input].delete(:selected)
        include_blank = options[:input].delete(:include_blank)

        @collection_options = {
          checked: [checked, selected, polymorphic_value, value].find { |value| value != nil },
          selected: ([selected, checked, polymorphic_value, value].find { |value| value != nil } unless kind_of?(Effective::FormInputs::Radios)),
          include_blank: include_blank
        }.compact
      end

      def html_options
        # Not sure why I need this. But they're merged in if I don't.
        options[:input].merge(skip_default_ids: nil, allow_method_names_outside_object: nil)
      end

      def value_method
        @value_method ||= options[:input].delete(:value_method)
      end

      def label_method
        @label_method ||= options[:input].delete(:label_method)
      end

      def group_method
        @group_method ||= options[:input].delete(:group_method)
      end

      def group_label_method
        @group_label_method ||= options[:input].delete(:group_label_method)
      end

      def option_key_method
        @option_key_method ||= options[:input].delete(:option_key_method)
      end

      def option_value_method
        @option_value_method ||= options[:input].delete(:option_value_method)
      end

      def options_collection
        @options_collection || []
      end

      # This is a grouped polymorphic collection
      # [["Clinics", [["Clinc 50", "Clinic_50"], ["Clinic 43", "Clinic_43"]]], ["Contacts", [["Contact 544", "Contact_544"]]]]
      def assign_options_collection!
        collection = options[:input].delete(:collection) || raise('Please include a collection')

        grouped = collection.kind_of?(Hash) && collection.values.first.respond_to?(:to_a)

        if grouped? && !grouped && collection.present?
          raise "Grouped collection expecting a Hash {'Posts' => Post.all, 'Events' => Event.all} or a Hash {'Posts' => [['Post A', 1], ['Post B', 2]], 'Events' => [['Event A', 1], ['Event B', 2]]}"
        end

        if polymorphic? && !grouped && collection.present?
          raise "Polymorphic collection expecting a Hash {'Posts' => Post.all, 'Events' => Event.all}"
        end

        @options_collection = (
          if polymorphic?
            collection.inject({}) { |h, (k, group)| h[k] = translate(group).map { |obj| [obj.to_s, "#{obj.class.model_name}_#{obj.id}"] }; h }
          elsif grouped
            collection.inject({}) { |h, (k, group)| h[k] = translate(group).map { |obj| obj }; h }
          elsif (collection == :boolean || collection == :booleans || collection == :boolean_collection)
            EffectiveBootstrap.boolean_collection
          else
            translate(collection).map { |obj| obj }
          end
        )
      end

      # Apply ActsAsArchived behavior. That's all for now.
      def translate(collection)
        return collection unless object.respond_to?(:new_record?)
        return collection unless collection.respond_to?(:klass)

        if collection.klass.respond_to?(:acts_as_archived?)
          collection = if object.new_record?
            collection.unarchived
          else
            collection.unarchived.or(collection.archived.where(collection.klass.primary_key => value))
          end
        end

        if respond_to?(:ajax?) && ajax? # effective_select
          collection = collection.where(collection.klass.primary_key => value)
        end

        collection
      end

      def assign_options_collection_methods!
        options[:input].reverse_merge!(
          if polymorphic?
            { group_method: :last, group_label_method: :first, option_key_method: :second, option_value_method: :first }
          elsif grouped?
            first = Array(options_collection.values.first).first

            string_of_strings = (
              options_collection.kind_of?(Hash) && 
              options_collection.keys.all? { |key| key.kind_of?(String) } && 
              options_collection.values.all? { |values| values.kind_of?(Array) && values.all? { |value| value.kind_of?(String) } }
            )

            if first.kind_of?(ActiveRecord::Base)
              { group_method: :last, group_label_method: :first, option_value_method: :to_s, option_key_method: :id }
            elsif string_of_strings
              { group_method: :last, group_label_method: :first, option_value_method: :to_s, option_key_method: :to_s }
            else
              { group_method: :last, group_label_method: :first, option_value_method: :first, option_key_method: :second }
            end
          elsif options_collection.first.kind_of?(Array)
            { label_method: :first, value_method: :second }
          elsif options_collection.first.kind_of?(ActiveRecord::Base)
            { label_method: :to_s, value_method: :id }
          else
            { label_method: :to_s, value_method: :to_s }
          end
        )
      end

      def polymorphic_type_method
        name.to_s.sub('_id', '') + '_type'
      end

      def polymorphic_id_method
        name.to_s.sub('_id', '') + '_id'
      end

      def polymorphic_value
        return nil unless polymorphic?
        "#{value.class.model_name}_#{value.id}" if value
      end

      def polymorphic_type_value
        value.try(:class).try(:model_name)
      end

      def polymorphic_id_value
        value.try(:id)
      end

    end
  end
end