Module: RakeCommander::Base::ClassHelpers
- Included in:
- ClassAutoLoader, Option, RakeTask::ClassMethods
- Defined in:
- lib/rake-commander/base/class_helpers.rb
Constant Summary collapse
- NOT_USED =
'no_used!'.freeze
Instance Method Summary collapse
-
#class_resolver(name, klass) ⇒ Object
Defines a class and instance method for lazy resolving a class.
-
#descendants(parent_class: self, direct: false, scope: nil) ⇒ Arrary<Class>
Finds all child classes of the current class.
-
#descendants?(parent_class: self, direct: false) ⇒ Boolean
trueif the current class has child classes, andfalseotherwise. -
#inheritable_attrs(*attrs, add_accessors: false) ⇒ Object
Builds the attr_reader and attr_writer of
attrsand registers the associated instance variable as inheritable. -
#inheritable_class_vars(*vars) ⇒ Object
Keeps track on class instance variables that should be inherited by child classes.
-
#inherited(subclass) ⇒ Object
This callback method is called whenever a subclass of the current class is created.
-
#instance_variable_name(name) ⇒ Object
Helper to create an instance variable
name. -
#new_class(name = "Child#{uid}", inherits: self, namespace: inherits) {|child_class| ... } ⇒ Class
If the class for
nameexists, it returns it. -
#redef_without_warning(const, value) ⇒ Object
Redefines constant
constwithvaluewithout triggering a warning. -
#resolve_class(klass, source_class: self, exception: true) ⇒ Class
Class resolver.
-
#to_constant(key) ⇒ String
Helper to normalize
keyinto a correctrubyconstant name. -
#used_param?(val) ⇒ Boolean
Helper to determine if a paramter has been used.
Instance Method Details
#class_resolver(name, klass) ⇒ Object
Defines a class and instance method for lazy resolving a class.
22 23 24 25 |
# File 'lib/rake-commander/base/class_helpers.rb', line 22 def class_resolver(name, klass) define_singleton_method(name) { resolve_class(klass) } define_method(name) { self.class.resolve_class(klass) } end |
#descendants(parent_class: self, direct: false, scope: nil) ⇒ Arrary<Class>
Finds all child classes of the current class.
103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 |
# File 'lib/rake-commander/base/class_helpers.rb', line 103 def descendants(parent_class: self, direct: false, scope: nil) scope ||= ObjectSpace.each_object(::Class) return [] if scope.empty? scope.select do |klass| klass < parent_class end.sort do |k_1, k_2| next -1 if k_2 < k_1 next 1 if k_1 < k_2 0 end.tap do |siblings| if direct siblings.reject! do |si| siblings.any? {|s| si < s} end end end end |
#descendants?(parent_class: self, direct: false) ⇒ Boolean
Returns true if the current class has child classes, and false otherwise.
124 125 126 |
# File 'lib/rake-commander/base/class_helpers.rb', line 124 def descendants?(parent_class: self, direct: false) !descendants(parent_class: parent_class, direct: direct).empty? end |
#inheritable_attrs(*attrs, add_accessors: false) ⇒ Object
Builds the attr_reader and attr_writer of attrs and registers the associated instance variable as inheritable.
140 141 142 143 144 145 146 147 148 149 |
# File 'lib/rake-commander/base/class_helpers.rb', line 140 def inheritable_attrs(*attrs, add_accessors: false) if add_accessors attrs.each do |attr| class_eval " class << self; attr_accessor :\#{attr} end\n RUBY\n end\n end\n inheritable_class_vars(*attrs)\nend\n", __FILE__, __LINE__ + 1 |
#inheritable_class_vars(*vars) ⇒ Object
- subclasses will inherit the value as is at that moment
- any change afterwards will be only on the specific class (in line with class instance variables)
- adapted from https://stackoverflow.com/a/10729812/4352306
Keeps track on class instance variables that should be inherited by child classes. TODO: this separates the logic of the method to the instance var. Think if would be possible to join them somehow.
134 135 136 137 |
# File 'lib/rake-commander/base/class_helpers.rb', line 134 def inheritable_class_vars(*vars) @inheritable_class_vars ||= [:inheritable_class_vars] @inheritable_class_vars += vars end |
#inherited(subclass) ⇒ Object
- values of the instance variables are copied as they are (no dups or clones)
- the above means: avoid methods that change the state of the mutable object on it
- mutating methods would reflect the changes on other classes as well
- therefore,
freezewill be called on the values that are inherited.
This callback method is called whenever a subclass of the current class is created.
157 158 159 160 161 162 163 164 165 |
# File 'lib/rake-commander/base/class_helpers.rb', line 157 def inherited(subclass) super.tap do inheritable_class_vars.each do |var| instance_var = instance_variable_name(var) value = instance_variable_get(instance_var) subclass.instance_variable_set(instance_var, value.freeze) end end end |
#instance_variable_name(name) ⇒ Object
Helper to create an instance variable name
71 72 73 74 75 |
# File 'lib/rake-commander/base/class_helpers.rb', line 71 def instance_variable_name(name) str = name.to_s str = "@#{str}" unless str.start_with?("@") str end |
#new_class(name = "Child#{uid}", inherits: self, namespace: inherits) {|child_class| ... } ⇒ Class
If the class for name exists, it returns it. Otherwise it generates it.
84 85 86 87 88 89 90 91 92 93 94 95 96 |
# File 'lib/rake-commander/base/class_helpers.rb', line 84 def new_class(name = "Child#{uid}", inherits: self, namespace: inherits) name = name.to_s.to_sym.freeze class_name = to_constant(name) unless target_class = resolve_class("#{namespace}::#{class_name}", exception: false) target_class = Class.new(inherits) Kernel.const_get(namespace.to_s).const_set class_name, target_class end target_class.tap do |klass| yield(klass) if block_given? end end |
#redef_without_warning(const, value) ⇒ Object
Redefines constant const with value without triggering a warning.
16 17 18 19 |
# File 'lib/rake-commander/base/class_helpers.rb', line 16 def redef_without_warning(const, value) self.class.send(:remove_const, const) if self.class.const_defined?(const) self.class.const_set(const, value) end |
#resolve_class(klass, source_class: self, exception: true) ⇒ Class
it caches the resolved klasses
Class resolver
34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 |
# File 'lib/rake-commander/base/class_helpers.rb', line 34 def resolve_class(klass, source_class: self, exception: true) @resolved ||= {} @resolved[klass] ||= case klass when Class klass when String begin Kernel.const_get(klass) rescue NameError raise if exception end when Symbol source_class.resolve_class(source_class.send(klass)) when Hash referrer, referred = klass.first resolve_class(referred, source_class: referrer, exception: exception) else raise "Unknown class: #{klass}" if exception end end |
#to_constant(key) ⇒ String
it removes namespace syntax ::
Helper to normalize key into a correct ruby constant name
60 61 62 63 64 65 66 |
# File 'lib/rake-commander/base/class_helpers.rb', line 60 def to_constant(key) key.to_s.strip.split(/::/).compact.map do |str| str.slice(0).upcase + str.slice(1..-1) end.join("").split(/[-_ :]+/i).compact.map do |str| str.slice(0).upcase + str.slice(1..-1) end.join("") end |
#used_param?(val) ⇒ Boolean
to effectivelly use this helper, you should initialize your target
paramters with the constant NOT_USED
Helper to determine if a paramter has been used
11 12 13 |
# File 'lib/rake-commander/base/class_helpers.rb', line 11 def used_param?(val) val != NOT_USED end |