class Tapioca::Dsl::Compilers::SmartProperties

: [ConstantType = singleton(::SmartProperties)]
~~~
end
end
def enabled=(enabled); end
sig { params(enabled: T.nilable(T::Boolean)).returns(T.nilable(T::Boolean)) }
def enabled; end
sig { returns(T.nilable(T::Boolean)) }
def published=(published); end
sig { params(published: T.nilable(T::Boolean)).returns(T.nilable(T::Boolean)) }
def published?; end
sig { returns(T.nilable(T::Boolean)) }
def description=(description); end
sig { params(description: ::String).returns(::String) }
def description; end
sig { returns(::String) }
def title=(title); end
sig { params(title: T.nilable(::String)).returns(T.nilable(::String)) }
def title; end
sig { returns(T.nilable(::String)) }
module SmartPropertiesGeneratedMethods
include SmartPropertiesGeneratedMethods
class Post
# typed: true
# post.rbi
~~~rbi
this compiler will produce the RBI file ‘post.rbi` with the following content:
~~~
end
property :enabled, accepts: [true, false], default: false
property :published, accepts: [true, false], reader: :published?
property! :description, accepts: String
property :title, accepts: String
include(SmartProperties)
class Post
# post.rb
~~~rb
For example, with the following class that includes `SmartProperties`:<br><br>(github.com/t6d/smart_properties).
`Tapioca::Dsl::Compilers::SmartProperties` generates RBI files for classes that include

def decorate

: -> void
@override
def decorate
  properties = T.unsafe(constant).properties #: ::SmartProperties::PropertyCollection
  return if properties.keys.empty?
  root.create_path(constant) do |k|
    smart_properties_methods_name = "SmartPropertiesGeneratedMethods"
    k.create_module(smart_properties_methods_name) do |mod|
      properties.values.each do |property|
        generate_methods_for_property(mod, property)
      end
    end
    k.create_include(smart_properties_methods_name)
  end
end

def gather_constants

: -> T::Enumerable[Module]
@override
def gather_constants
  all_modules.select do |c|
    name_of(c) &&
      ::SmartProperties > c &&
      ::SmartProperties::Validations::Ancestor != c &&
      ::SmartProperties::ClassMethods === c
  end
end

def generate_methods_for_property(mod, property)

: (RBI::Scope mod, ::SmartProperties::Property property) -> void
def generate_methods_for_property(mod, property)
  type = type_for(property)
  if property.writable?
    name = property.name.to_s
    method_name = "#{name}="
    mod.create_method(method_name, parameters: [create_param(name, type: type)], return_type: type)
  end
  mod.create_method(property.reader.to_s, return_type: type)
end

def type_for(property)

: (::SmartProperties::Property property) -> String
def type_for(property)
  converter, accepter, required = property.to_h.fetch_values(
    :converter,
    :accepter,
    :required,
  )
  return "T.untyped" if converter
  type = if accepter.nil? || accepter.respond_to?(:to_proc)
    "T.untyped"
  elsif accepter == Array
    "T::Array[T.untyped]"
  elsif BOOLEANS.include?(accepter)
    "T::Boolean"
  elsif Array(accepter).all? { |a| a.is_a?(Module) }
    accepters = Array(accepter)
    types = accepters.map { |mod| T.must(qualified_name_of(mod)) }.join(", ")
    types = "T.any(#{types})" if accepters.size > 1
    types
  else
    "T.untyped"
  end
  might_be_optional = Proc === required || !required
  type = as_nilable_type(type) if might_be_optional
  type
end