All Projects → joshleblanc → View_component_reflex

joshleblanc / View_component_reflex

Licence: mit

Programming Languages

ruby
36898 projects - #4 most used programming language

Projects that are alternatives of or similar to View component reflex

Front End Performance Checklist
🎮 더 빠르게 작동하는 프론트엔드 성능 체크리스트
Stars: ✭ 183 (-1.08%)
Mutual labels:  hacktoberfest
Yii2 Gii
Yii 2 Gii Extension
Stars: ✭ 183 (-1.08%)
Mutual labels:  hacktoberfest
Linebot
🤖 SDK for the LINE Messaging API for Node.js
Stars: ✭ 184 (-0.54%)
Mutual labels:  hacktoberfest
Tox
Command line driven CI frontend and development task automation tool.
Stars: ✭ 2,523 (+1263.78%)
Mutual labels:  hacktoberfest
Px4 Sitl gazebo
Set of plugins, models and worlds to use with OSRF Gazebo Simulator in SITL and HITL.
Stars: ✭ 182 (-1.62%)
Mutual labels:  hacktoberfest
Bundler Leak
Known-leaky gems verification for bundler: `bundle leak` to check your app and find leaky gems in your Gemfile 💎💧
Stars: ✭ 184 (-0.54%)
Mutual labels:  hacktoberfest
Slurp
Evaluate the security of S3 buckets
Stars: ✭ 183 (-1.08%)
Mutual labels:  hacktoberfest
Optaplanner
AI constraint solver in Java to optimize the vehicle routing problem, employee rostering, task assignment, maintenance scheduling, conference scheduling and other planning problems.
Stars: ✭ 2,454 (+1226.49%)
Mutual labels:  hacktoberfest
Docker
🐳
Stars: ✭ 183 (-1.08%)
Mutual labels:  hacktoberfest
Demo
Demo app for the API Platform framework
Stars: ✭ 184 (-0.54%)
Mutual labels:  hacktoberfest
Comunica
📬 A knowledge graph querying framework for JavaScript
Stars: ✭ 183 (-1.08%)
Mutual labels:  hacktoberfest
Terraform Provider Sentry
Terraform provider for Sentry
Stars: ✭ 183 (-1.08%)
Mutual labels:  hacktoberfest
Magento2 Menu
Provides powerful menu editor to replace category based menus in Magento 2
Stars: ✭ 184 (-0.54%)
Mutual labels:  hacktoberfest
Bastion
Highly-available Distributed Fault-tolerant Runtime
Stars: ✭ 2,333 (+1161.08%)
Mutual labels:  hacktoberfest
Training Material
A collection of Galaxy-related training material
Stars: ✭ 184 (-0.54%)
Mutual labels:  hacktoberfest
Goplaxt
Scrobble Plex plays to Trakt with ease!
Stars: ✭ 183 (-1.08%)
Mutual labels:  hacktoberfest
Chef Client
Development repository for Chef Client cookbook
Stars: ✭ 183 (-1.08%)
Mutual labels:  hacktoberfest
Discordchatexporter
Exports Discord chat logs to a file
Stars: ✭ 3,198 (+1628.65%)
Mutual labels:  hacktoberfest
Doctree
Repository of Japanese Ruby reference manual
Stars: ✭ 184 (-0.54%)
Mutual labels:  hacktoberfest
Pretty Simple
pretty-printer for Haskell data types that have a Show instance
Stars: ✭ 183 (-1.08%)
Mutual labels:  hacktoberfest

ViewComponentReflex

ViewComponentReflex allows you to write reflexes right in your view component code.

It builds upon stimulus_reflex and view_component

Usage

You can add reflexes to your component by inheriting from ViewComponentReflex::Component.

This will act as if you created a reflex with the method my_cool_stuff. To call this reflex, add data-reflex="click->MyComponentReflex#my_cool_reflex", just like you're using stimulus reflex.

ViewComponentReflex will maintain your component's instance variables between renders. You need to include data-key=<%= key %> on your root element, as well as any element that stimulates a reflex. ViewComponent is inherently state-less, so the key is used to reconcile state to its respective component.

Example

# counter_component.rb
class CounterComponent < ViewComponentReflex::Component
  def initialize
    @count = 0
  end

  def increment
    @count += 1
  end
end
# counter_component.html.erb
<%= component_controller do %>
    <p><%= @count %></p>
    <%= reflex_tag :increment, :button, "Click" %>
<% end %>

Collections

In order to reconcile state to components in collections, you can specify a collection_key method that returns some value unique to that component.

class TodoComponent < ViewComponentReflex::Component
  def initialize(todo:)
    @todo = todo
  end

  def collection_key
    @todo.id
  end
end
#
<%= render(TodoComponent.with_collection(Todo.all)) %>

API

permit_parameter?(initial_param, new_params)

If a new parameter is passed to the component during rendering, it is used instead of what's in state. If you're storing instances in state, you can use this to properly compare them.

def permit_parameter?(initial_param, new_param)
  if new_param.instance_of? MyModel 
    new_param.id == @my_model.id
  else
    super
  end
end

omitted_from_state

Return an array of instance variables you want to omit from state. Only really useful if you're using the session state adapter, and you have an instance variable that can't be serialized.

def omitted_from_state
  [:@form]
end

reflex_tag(reflex, name, content_or_options_with_block = nil, options = nil, escape = true, &block)

This shares the same definition as content_tag, except it accepts a reflex as the first parameter.

<%= reflex_tag :increment, :button, "Click me!" %>

Would add a click handler to the increment method on your component.

To use a non-click event, specific that with -> notation

<%= reflex_tag "mouseenter->increment", :button, "Click me!" %>

reflex_data_attributes(reflex)

This helper will give you the data attributes used in the reflex_tag above if you want to build your own elements.

Build your own tag:

<%= link_to (image_tag photo.image.url(:medium)), data: reflex_data_attributes(:increment) %>

Render a ViewComponent

<%= render ButtonComponent.new(data: reflex_data_attributes("mouseenter->increment")) %>

Make sure that you assign the reflex_data_attributes to the correct element in your component.

collection_key

If you're rendering a component as a collection with MyComponent.with_collection(SomeCollection), you must define this method to return some unique value for the component. This is used to reconcile state in the background.

def initialize
  @my_model = MyModel.new
end

def collection_key
  @my_model.id
end

stimulate(target, data)

Stimulate another reflex from within your component. This typically requires the key of the component you're stimulating which can be passed in via parameters.

def initialize(parent_key)
  @parent_key = parent_key
end

def stimulate_other
  stimulate("OtherComponent#method", { key: @parent_key })
end

refresh!(selectors)

Refresh a specific element on the page. Using this will implicitly run prevent_render!. If you want to render a specific element, as well as the component, a common pattern would be to pass selector as one of the parameters

def my_method
  refresh! '#my-special-element', selector
end

selector

Returns the unique selector for this component. Useful to pass to refresh! when refreshing custom elements.

prevent_refresh!

By default, VCR will re-render your component after it executes your method. prevent_refresh! prevents this from happening.

def my_method
  prevent_refresh!
  @foo = :bar
end # the rendered page will not reflect this change

refresh_all!

Refresh the entire body of the page

def do_some_global_action
  prevent_refresh!
  session[:model] = MyModel.new
  refresh_all!
end

stream_to(channel)

Stream to a custom channel, rather than the default stimulus reflex one

def do_something
  stream_to MyChannel
  
  @foo = :bar
end

key

This is a key unique to a particular component. It's used to reconcile state between renders, and should be passed as a data attribute whenever a reflex is called

<button type="button" data-reflex="click->MyComponent#do_something" data-key="<%= key %>">Click me!</button>

component_controller(options = {}, &blk)

This is a view helper to properly connect VCR to the component. It outputs <div data-controller="my-controller" key=<%= key %></div> You must wrap your component in this for everything to work properly.

<%= component_controller do %>
  <p><%= @count %></p
<% end %>

after_state_initialized(parameters_changed)

This is called after the state has been inserted in the component. You can use this to run conditional functions after some parameter has superseeded whatever's in state

def after_state_initialized(parameters_changed)
  if parameters_changed.include?(:@filter)
    calculate_visible_rows
  end
end

Custom reflex base class

Reflexes typically inherit from a base ApplicationReflex. You can define the base class for a view_component_reflex by using the reflex_base_class accessor. The parent class must inherit ViewComponentReflex::Reflex, and will throw an error if it does not.

class ApplicationReflex < ViewComponentReflex::Reflex

end


class MyComponent < ViewComponentReflex::Component
  MyComponent.reflex_base_class = ApplicationReflex
end

Common patterns

A lot of the time, you only need to update specific components when changing instance variables. For example, changing @loading might only need to display a spinner somewhere on the page. You can define setters to implicitly render the appropriate pieces of dom whenever that variable is set

def initialize
  @loading = false
end

def loading=(new_value)
  @loading = new_value
  refresh! '#loader'
end

def do_expensive_action
  prevent_refresh! 

  self.loading = true
  execute_it
  self.loading = false
end
<%= component_controller do %>
  <div id="loader"> 
    <% if @loading %>
      <p>Loading...</p>
    <% end %>
  </div>

  <button type="button" data-reflex="click->MyComponent#do_expensive_action" data-key="<%= key %>">Click me!</button>
<% end

State

By default (since version 2.3.2), view_component_reflex stores component state in session. You can optionally set the state adapter to use the memory by changing config.state_adapter to ViewComponentReflex::StateAdapter::Memory.

Custom State Adapters

ViewComponentReflex uses session for its state by default. To change this, add an initializer to config/initializers/view_component_reflex.rb.

ViewComponentReflex::Engine.configure do |config|
  config.state_adapter = YourAdapter
end

Existing Fast Redis based State Adapter

This adapter uses hmset and hgetall to reduce the number of operations. This is the recommended adapter if you are using AnyCable.

ViewComponentReflex::Engine.configure do |config|
  config.state_adapter = ViewComponentReflex::StateAdapter::Redis.new(
      redis_opts: {
          url: "redis://localhost:6379/1", driver: :hiredis
      },
      ttl: 3600)
end

YourAdapter should implement

class YourAdapter
  ##
  # request - a rails request object
  # key - a unique string that identifies the component instance
  def self.state(request, key)
    # Return state for a given key
  end

  ##
  # set_state is used to modify the state.
  #
  # request - a rails request object
  # controller - the current controller
  # key - a unique string that identifies the component
  # new_state - the new state to set
  def self.set_state(request, controller, key, new_state)
    # update the state
  end


  ##
  # store_state is used to replace the state entirely. It only accepts
  # a request object, rather than a reflex because it's called from the component's 
  # side with the component's instance variables.
  #
  # request - a rails request object
  # key - a unique string that identifies the component instance
  # new_state - a hash containing the component state
  def self.store_state(request, key, new_state = {})
    # replace the state
  end
end

Installation

Add this line to your application's Gemfile:

gem 'view_component_reflex'

And then execute:

$ bundle

Or install it yourself as:

$ gem install view_component_reflex

Common problems

Uninitialized constants <component>Reflex

A component needs to be wrapped in <%= component_controller do %> in order to properly initialize, otherwise the Reflex class won't get created.

Session is an empty hash

StimulusReflex 3.3 introduced selector morphs, allowing you to render arbitrary strings via ApplicationController.render, for example:

def test_selector
  morph '#some-container', ApplicationController.render(MyComponent.new(some: :param))
end

StimulusReflex 3.4 introduced a fix that merges the current request.env and provides the CSRF token to fetch the session.

Help, my instance variables do not persist into the session

These instance variable names are not working and unsafe:

def unsafe_instance_variables
  [
    :@view_context, :@lookup_context, :@view_renderer, :@view_flow,
    :@virtual_path, :@variant, :@current_template, :@output_buffer, :@key,
    :@helpers, :@controller, :@request, :@tag_builder, :@initialized_state
  ]
end

Please use a different name to be able to save them to the session.

Anycable

@sebyx07 provided a solution to use anycable (https://github.com/joshleblanc/view_component_reflex/issues/23#issue-721786338)

Leaving this, might help others:

I tried this with any cable and I had to add this to development.rb Otherwise @instance_variables were nil after a reflex

  config.cache_store = :redis_cache_store, { url: "redis://localhost:6379/1", driver: :hiredis }
  config.session_store :cache_store

License

The gem is available as open source under the terms of the MIT License.

Caveats

State uses session to maintain state as of right now. It also assumes your component view is written with a file extension of either .html.erb, .html.haml or .html.slim.

Support me

Buy Me A Coffee

Note that the project description data, including the texts, logos, images, and/or trademarks, for each open source project belongs to its rightful owner. If you wish to add or remove any projects, please contact us at [email protected].