Skip to content

Latest commit

 

History

History
153 lines (121 loc) · 5.7 KB

customizing_view_components.md

File metadata and controls

153 lines (121 loc) · 5.7 KB

Customizing view components

Solidus Admin uses view components to render the views. Components are a pattern for breaking up the view layer into small, reusable pieces, easy to reason about and test.

All the components Solidus Admin uses are located in the app/components folder of the solidus_admin gem. As you can see, they are organized in a particular folder structure:

  • All of them are under the SolidusAdmin namespace.
  • They are grouped in sidecar directories, where the main component file and all its related files (assets, i18n files, etc.) live together.

For instance, the component for the main navigation is located in app/components/solidus_admin/main_nav/component.rb.

Solidus Admin components are designed to be easily customizable by the host application. Because of that, if you look at how they are designed, you'll find a series of patterns are followed consistently:

  • Components are always resolved from a global registry in SolidusAdmin::Config.components instead of being referenced by their constant. For example, we call component('main_nav') instead of referencing SolidusAdmin::MainNav::Component directly. As you'll see later, this makes it easy to replace or tweak a component.
  • It's possible to override the registry locally inside a component by redefining the #component method. This is useful when you need to use a component that is not registered in the global registry or need to address some edge case.
  • In any case, Solidus Admin components initializers only take keyword arguments.

A picture is worth a thousand words, so let's depict how this works with an example:

# app/components/solidus_admin/foo/component.rb
class SolidusAdmin::Foo::Component < SolidusAdmin::BaseComponent
  def component(key)
    return MyApplication::Bar::Component if key == 'bar'

    super
  end

  erb_template <<~ERB
    <div>
      <%= render component('bar').new %>
    </div>
  ERB
end
# render component('foo').new

Customizing components

Some of the customizations detailed below require you to match Solidus Admin's component paths in your application. For instance, if we talk about the component in solidus_admin gem's app/components/solidus_admin/main_nav/component.rb, the matching path in your application would be app/components/my_application/solidus_admin/main_nav/component.rb, where my_application is the underscored name of your application (you can get it by running Rails.application.class.module_parent_name).

Replacing a component's template

In the most typical case, you'll only need to replace the template used by a component. You can do that by creating a new component with a maching path in your application, inheriting from the default one. Then, you can create a new template for it. For example, to replace the main nav template:

# app/components/my_application/solidus_admin/main_nav/component.rb %>
class MyApplication::SolidusAdmin::MainNav::Component < ::SolidusAdmin::MainNav::Component
end

<%# app/components/my_application/solidus_admin/main_nav/component.html.erb %>
<nav class="my_own_classes">
  <%=
    render main_nav_item_component.with_collection(
      sorted_items
    )
  %>
</nav>

Prepending or appending to a component's template

In some situations, you might only need to add some markup before or after a component. You can easily do that by rendering the Solidus Admin component and adding your markup before or after it.

<%# app/components/my_application/solidus_admin/main_nav/component.html.erb %>
<h1>MY STORE ADMINISTRATION</h1>
<%= render SolidusAdmin::MainNav::Component.new %>

Replacing a component

You can replace a component by creating a new one with a matching path in your application.

There are two considerations to keep in mind:

  • Be aware that other components might be using the component you're replacing. They should only be using its #initialize method, so make sure to keep compatibility with it when they're called.
  • Solidus Admin's components always inherit from SolidusAdmin::BaseComponent. You can consider doing the same if you need to use one of its helpers.

For example, the following replaces the main nav component:

# app/components/my_application/solidus_admin/main_nav/component.rb
class MyApplication::SolidusAdmin::MainNav::Component < SolidusAdmin::BaseComponent
  # do your thing
end

If you need more control, you can explicitly register your component in the Solidus Admin container instead of using an implicit path:

ⓘ Right now, that will raise an error when the application is reloaded. We need to fix it.

# config/initalizers/solidus_admin.rb
Rails.application.config.to_prepare do
  SolidusAdmin::Config.components['ui/button'] = MyApplication::Button::Component
end

Tweaking a component

If you only need to tweak a component, you can always inherit from it in a matching path from within your application (or manually add it to the registry) and override the methods you need to change:

# app/components/my_application/solidus_admin/main_nav/component.rb
class MyApplication::SolidusAdmin::MainNav::Component < ::SolidusAdmin::MainNav::Component
  def sorted_items
    super.reverse
  end
end

Be aware that this approach comes with an important trade-off: You'll need to keep your component in sync with the original one as it changes on future updates. For instance, in the example above, the component is overriding a private method, so there's no guarantee that it will continue to exist in the future without being deprecated, as we only guarantee public API stability.