lib/icalendar/has_components.rb



# frozen_string_literal: true

module Icalendar

  module HasComponents

    def self.included(base)
      base.extend ClassMethods
      base.class_eval do
        attr_reader :custom_components
      end
    end

    def initialize(*args)
      @custom_components = Hash.new
      super
    end

    def add_component(c)
      c.parent = self
      yield c if block_given?
      send("#{c.name.downcase}s") << c
      c
    end

    def add_custom_component(component_name, c)
      c.parent = self
      yield c if block_given?
      (custom_components[component_name.downcase.gsub("-", "_")] ||= []) << c
      c
    end

    def custom_component(component_name)
      custom_components[component_name.downcase.gsub("-", "_")] || []
    end

    METHOD_MISSING_ADD_REGEX = /^add_(x_\w+)$/.freeze
    METHOD_MISSING_X_FLAG_REGEX = /^x_/.freeze

    def method_missing(method, *args, &block)
      method_name = method.to_s
      if method_name =~ METHOD_MISSING_ADD_REGEX
        component_name = $1
        custom = args.first || Component.new(component_name, component_name.upcase)
        add_custom_component(component_name, custom, &block)
      elsif method_name =~ METHOD_MISSING_X_FLAG_REGEX && custom_component(method_name).size > 0
        custom_component method_name
      else
        super
      end
    end

    def respond_to_missing?(method_name, include_private = false)
      string_method = method_name.to_s
      string_method.start_with?('add_x_') || custom_component(string_method).size > 0 || super
    end

    module ClassMethods
      def components
        @components ||= []
      end

      def component(singular_name, find_by = :uid, klass = nil)
        components = "#{singular_name}s"
        self.components << components
        component_var = "@#{components}"

        define_method components do
          if instance_variable_defined? component_var
            instance_variable_get component_var
          else
            instance_variable_set component_var, []
          end
        end

        define_method singular_name do |c = nil, &block|
          if c.nil?
            c = begin
              klass ||= Icalendar.const_get singular_name.capitalize
              klass.new
            rescue NameError => ne
              Icalendar.logger.warn ne.message
              Component.new singular_name
            end
          end

          add_component c, &block
        end

        define_method "find_#{singular_name}" do |id|
          send(components).find { |c| c.send(find_by) == id }
        end if find_by

        define_method "add_#{singular_name}" do |c|
          send singular_name, c
        end

        define_method "has_#{singular_name}?" do
          !send(components).empty?
        end
      end
    end
  end

end