module Preact::ComponentResolution def self.included(base) base.instance_exec do unless method_defined?(:_preact_component_class_resolution_original_const_missing) alias _preact_component_class_resolution_original_const_missing const_missing end def const_missing(const_name) %x{ if (typeof Opal.global[const_name] !== "undefined" && (const_name[0] === const_name[0].toUpperCase())) { var new_const = Opal.Preact.NativeConstantWrapper.$new(Opal.global[const_name], const_name); new_const.preact_component = Opal.global[const_name]; Opal.Object.$const_set(const_name, new_const); return new_const; } else { return #{_preact_component_class_resolution_original_const_missing(const_name)}; } } end # this is required for autoloading support, as the component may not be loaded and so its method is not registered. # must load it first, done by const_get, and next time the method will be there. unless method_defined?(:_preact_component_class_resolution_original_method_missing) alias _preact_component_class_resolution_original_method_missing method_missing end def method_missing(component_name, *args, &block) # check for ruby component and render it # otherwise pass on method missing %x{ var constant = null; if (typeof self.iso_preact_const_cache === 'undefined') { self.iso_preact_const_cache = {}; } if (typeof self.iso_preact_const_cache[component_name] !== 'undefined') { constant = self.iso_preact_const_cache[component_name] } else { try { constant = self.$const_get(component_name); self.iso_preact_const_cache[component_name] = constant; } catch(err) { } } if (constant && typeof constant.preact_component !== 'undefined') { return Opal.Preact.internal_prepare_args_and_render(constant.preact_component, args, block); } return #{_preact_component_class_resolution_original_method_missing(component_name, *args, block)}; } end end end unless method_defined?(:_preact_component_resolution_original_method_missing) alias _preact_component_resolution_original_method_missing method_missing end def method_missing(component_name, *args, &block) # Further on it must check for modules, because $const_get does not take # the full nesting into account, as usually its called via $$ with the # nesting provided by the compiler. %x{ var constant; if (typeof self.iso_preact_const_cache === 'undefined') { self.iso_preact_const_cache = {}; } if (typeof self.iso_preact_const_cache[component_name] !== 'undefined') { constant = self.iso_preact_const_cache[component_name] } else if (typeof self.$$is_a_module !== 'undefined') { try { constant = self.$const_get(component_name); self.iso_preact_const_cache[component_name] = constant; } catch(err) { } } else { let sc = self.$class(); try { constant = sc.$const_get(component_name); self.iso_preact_const_cache[component_name] = constant; } catch(err) { var module_names = sc.$to_s().split("::"); var module_name; for (var i = module_names.length - 1; i > 0; i--) { module_name = module_names.slice(0, i).join('::'); try { constant = sc.$const_get(module_name).$const_get(component_name, false); self.iso_preact_const_cache[component_name] = constant; break; } catch(err) { } } } } if (constant && typeof constant.preact_component !== 'undefined') { return Opal.Preact.internal_prepare_args_and_render(constant.preact_component, args, block); } return #{_preact_component_resolution_original_method_missing(component_name, *args, block)}; } end end