Upgrading to Phlex v2
While we’ve tried to keep breaking changes to a minimum, there are a few things you will need to be aware of when upgrading from Phlex v1 to v2.
The latest version of v1 contains a number of deprecations, so we recommend upgrading to the latest version of v1 first.
Dropping SemVer for BreakVer
Phlex v1 used SemVer (semantic versioning). Going forward, Phlex v2 and up will use BreakVer instead. BreakVer will allow us to release changes more frequently because we can distinguish between major breaking changes and minor breaking changes.
The version scheme is:
MAJOR . MINOR . NON_BREAKINGNON_BREAKINGchanges should always be safe to apply. They can include new features, enhancements, refactors and bug fixes, but they should never include any breaking changes.MINORchanges might break code in a minor way — usually in a way that can be easily accommodated with a few minutes of mostly find/replace-type work.MAJORchanges might break code in a major way. These are milestone releases.
We try to avoid breaking changes altogether, but they are sometimes necessary for progress. We’ve come to realize if we’re going to make a breaking change, we should make it as quickly as possible. Any delay means more work for users.
Kits new
Originally previewed in v1, Kits are now out of beta and fully supported in v2. Kits are a way to package up a set of components into a module, which makes them easier to render.
In v2, Kits also extend to modules (but not classes) defined under them.
module Components
extend Phlex::Kit
module Articles
# this is automatically upgraded to a kit
class List < Phlex::HTML
# this is available on the `Components::Articles` kit
end
end
endA better attribute cache new
Phlex v2 introduces a new attribute cache that caches more things.
Renamed template → view_template breaking
Instead of defining the template method for your component templates, you should instead define view_template. This was renamed so that the template method can be used for <template> HTML tags.
Renamed template_tag → template breaking
To render <template> elements in a Phlex::HTML component, you need to call the template method instead of template_tag.
Made yield_content private
You can replace it with render, which now accepts blocks.
Removed tokens and classes breaking
There are better ways to handle conditional tokens now, so we removed these helpers. If you need them back to support your existing code, you can just copy the original implementation from below.
Original classes and tokens implementation
def classes(*tokens, **conditional_tokens)
tokens = self.tokens(*tokens, **conditional_tokens)
if tokens.empty?
{}
else
{ class: tokens }
end
end
def tokens(*tokens, **conditional_tokens)
conditional_tokens.each do |condition, token|
truthy = case condition
when Symbol then send(condition)
when Proc then condition.call
else raise ArgumentError, "The class condition must be a Symbol or a Proc."
end
if truthy
case token
when Hash then __append_token__(tokens, token[:then])
else __append_token__(tokens, token)
end
else
case token
when Hash then __append_token__(tokens, token[:else])
end
end
end
tokens = tokens.select(&:itself).join(" ")
tokens.strip!
tokens.gsub!(/\s+/, " ")
tokens
end
private
def __append_token__(tokens, token)
case token
when nil then nil
when String then tokens << token
when Symbol then tokens << token.name
when Array then tokens.concat(token)
else raise ArgumentError,
"Conditional classes must be Symbols, Strings, or Arrays of Symbols or Strings."
end
endRenamed unsafe_raw → raw breaking
We’ve renamed unsafe_raw to raw, and it will now only output strings that are branded as being HTML-safe. You can use the new safe helper to mark content as safe. If you’re using Rails, ActiveSupport::SafeBuffer is also treated as safe.
def view_template
rendered_markdown = Commonmarker.to_html(@markdown)
unsafe_raw rendered_markdown
rendered_markdown = safe Commonmarker.to_html(@markdown)
raw rendered_markdown
endRemoved DeferredRender breaking
DeferredRender was an odd combination of something that is easy to implement and hard to explain. We decided to remove it as a feature so that we don’t have to explain it. 😇
You can recreate the effect DeferredRender had with this module.
module DeferredRender
def before_template(&)
vanish(&)
super
end
endSee Yielding for an explanation how how this works.
Changed selective rendering breaking
We’ve redesigned the Selective Rendering feature (introduced in 1.10) to be more predictable and easier to understand.
Previously, selective rendering worked by targeting element IDs:
# Before (Phlex ~> 1.10)
def view_template
section do
ul(id: "the-list") do # Could target this by ID
li { "Item 1" }
li { "Item 2" }
end
end
end
# Usage:
component.call(fragments: ["the-list"])Now, selective rendering requires explicit fragment declarations:
# After (Phlex 2.0)
def view_template
section do
fragment("the-list") do # Explicitly declare renderable fragment
ul(id: "the-list") do
li { "Item 1" }
li { "Item 2" }
end
end
end
end
# Usage remains the same:
component.call(fragments: ["the-list"])Key Differences:
- Explicit Fragment Declaration: Only content wrapped in
fragment(name) { ... }can be selectively rendered - Decoupled from DOM: Fragment names no longer need to match element IDs
- More Predictable: Eliminates edge cases where ID-based targeting wasn’t supported
Common Use Case: Turbo Frames
For applications using Turbo Frames, you can automatically make all frames selectively renderable by extending the turbo_frame method:
def turbo_frame(id:, ...)
fragment(id) { super }
endThis ensures any <turbo-frame> element can be selectively rendered using its ID.
New opinionated Rails generators breaking
We’ve made some significant changes to the Rails generators, which now assume a specific folder structure and naming convention for views and components.
The install generator now create an initializer file in config/initializers/phlex.rb where the modules: Views and Components are defined. It also autoloads the app/views and app/components directories with the Views and Components namespaces respectively.
config/initializers/phlex.rb
# frozen_string_literal: true
module Views
end
module Components
extend Phlex::Kit
end
Rails.autoloaders.main.push_dir(
"#{Rails.root}/app/views", namespace: Views
)
Rails.autoloaders.main.push_dir(
"#{Rails.root}/app/components", namespace: Components
)It also creates a Base class for both views and components.
The view generator now creates views under app/views, namespaced under Views.
Example view
# frozen_string_literal: true
class Views::Articles::Index < Views::Base
def view_template
h1 { "Articles" }
end
endThe component generator now creates components under app/components, namespaced under Components.
Example component
# frozen_string_literal: true
class Components::Button < Components::Base
def view_template
button { "Click me" }
end
endYou may want to run the new install generator in a fresh Rails app to see how the new folder structure works and assess if you want to adopt it.
The folder structure is entirely optional — you can put Phlex components wherever you like — but guides and generators may assume this structure.
Changed rendering partials from Phlex Railsbreaking
In order to support rendering plain strings, we removed the ability to render Rails partials like this:
render "foo"Now, you must use the partial method to create a partial reference object.
render partial("foo")These partial reference objects are also renderable outside of Phlex. You can pass them to ViewComponent components or other Rails partials.