require 'hyperstack/ext/component/string' require 'hyperstack/ext/component/hash' require 'active_support/core_ext/class/attribute' require 'hyperstack/internal/auto_unmount' require 'hyperstack/internal/component/rendering_context' require 'hyperstack/internal/component' require 'hyperstack/internal/component/instance_methods' require 'hyperstack/internal/component/class_methods' require 'hyperstack/internal/component/props_wrapper' module Hyperstack module Component def self.included(base) base.include(Hyperstack::State::Observer) base.include(Hyperstack::Internal::Component::InstanceMethods) base.include(Hyperstack::Internal::AutoUnmount) # pulls in the CallBacks module as well base.include(Hyperstack::Internal::Component::Tags) base.include(Hyperstack::Internal::Component::ShouldComponentUpdate) base.class_eval do class_attribute :initial_state define_callback :before_mount define_callback :after_mount define_callback :before_new_params define_callback :before_update define_callback :after_update define_callback :__hyperstack_component_after_render_hook define_callback :__hyperstack_component_rescue_hook #define_callback :before_unmount defined already by Async module define_callback(:after_error) { Hyperstack::Internal::Component::ReactWrapper.add_after_error_hook(base) } end base.extend(Hyperstack::Internal::Component::ClassMethods) unless `Opal.__hyperstack_component_original_defn` %x{ Opal.__hyperstack_component_original_defn = Opal.defn Opal.defn = function(klass, name, fn) { #{ if `klass`.respond_to?(:hyper_component?) && !(`klass` < Hyperstack::Component::NativeLibrary) if `name == '$render'` && !`klass`.allow_deprecated_render_definition? Hyperstack.deprecation_warning(`klass`, 'Do not directly define the render method. Use the render macro instead.') elsif `name == '$__hyperstack_component_render'` `name = '$render'` end end } Opal.__hyperstack_component_original_defn(klass, name, fn) } } nil end end def self.mounted_components @__hyperstack_component_mounted_components ||= Set.new end def self.force_update! components = mounted_components.to_a # need a copy as force_update may change the list components.each do |comp| # check if its still mounted next unless mounted_components.include? comp comp.force_update! end end def self.deprecation_warning(message) Hyperstack.deprecation_warning(name, message) end def deprecation_warning(message) Hyperstack.deprecation_warning(self.class.name, message) end def initialize(native_element) @__hyperstack_component_native = native_element end def emit(event_name, *args) if Event::BUILT_IN_EVENTS.include?(built_in_event_name = "on#{event_name.to_s.event_camelize}") params[built_in_event_name].call(*args) else params["on_#{event_name}"].call(*args) end end def component_will_mount @__hyperstack_component_params_wrapper = self.class.props_wrapper.new(self) IsomorphicHelpers.load_context(true) if IsomorphicHelpers.on_opal_client? observing(immediate_update: true) do Hyperstack::Component.mounted_components << self run_callback(:before_mount, props) end end def component_did_mount observing(update_objects: true) do run_callback(:after_mount) Hyperstack::Internal::Component::RenderingContext.quiet_test(self) end end def component_will_receive_props(next_props) # need to rethink how this works in opal-react, or if its actually that useful within the react.rb environment # for now we are just using it to clear processed_params observing(immediate_update: true) { run_callback(:before_new_params, next_props) } @__hyperstack_component_receiving_props = true end def component_will_update(next_props, next_state) observing { run_callback(:before_update, next_props, next_state) } if @__hyperstack_component_receiving_props @__hyperstack_component_params_wrapper.reload(next_props) end @__hyperstack_component_receiving_props = false end def component_did_update(prev_props, prev_state) observing(update_objects: true) do run_callback(:after_update, prev_props, prev_state) Hyperstack::Internal::Component::RenderingContext.quiet_test(self) end end def component_will_unmount observing do unmount # runs unmount callbacks as well remove Hyperstack::Component.mounted_components.delete self end end def component_did_catch(error, info) observing do run_callback(:after_error, error, info) end end def mutations(_objects) # if we have to we may have to require that all objects respond to a "name" method (see legacy method update_react_js_state below) set_state('***_state_updated_at-***' => `Date.now() + Math.random()`) end def update_react_js_state(object, name, value) if object name = "#{object.class}.#{name}" unless object == self # Date.now() has only millisecond precision, if several notifications of # observer happen within a millisecond, updates may get lost. # to mitigate this the Math.random() appends some random number # this way notifactions will happen as expected by the rest of hyperloop set_state( '***_state_updated_at-***' => `Date.now() + Math.random()`, name => value ) else set_state name => value end end def render raise 'no render defined' end unless method_defined?(:render) def waiting_on_resources @__hyperstack_component_waiting_on_resources end def __hyperstack_component_run_post_render_hooks(element) run_callback(:__hyperstack_component_after_render_hook, element) { |*args| args }.first end def _render_wrapper observing(rendering: true) do element = Hyperstack::Internal::Component::RenderingContext.render(nil) do render || '' end @__hyperstack_component_waiting_on_resources = element.waiting_on_resources if element.respond_to? :waiting_on_resources element end end end end