style_capsule
CSS scoping extension for Ruby components. Provides attribute-based style encapsulation for Phlex, ViewComponent, and ERB templates to prevent style leakage between components. Works with Rails and can be used standalone in other Ruby frameworks (Sinatra, Hanami, etc.) or plain Ruby scripts. Includes configurable caching strategies for optimal performance.
Sponsored by Kisko Labs.
Installation
Add to your Gemfile:
gem "style_capsule"Then run bundle install.
Features
- Attribute-based CSS scoping (no class name renaming)
- Phlex, ViewComponent, and ERB support with automatic Rails integration
- Per-component-type scope IDs (shared across instances)
- CSS Nesting support (optional, more performant, requires modern browsers)
- Stylesheet registry with thread-safe head rendering, namespace support, and compatibility with Propshaft and other asset bundlers
- Multiple cache strategies: none, time-based, custom proc, and file-based (HTTP caching)
- Security protections: path traversal protection, input validation, size limits
Usage
Phlex Components
class MyComponent < ApplicationComponent
include StyleCapsule::Component
def component_styles
<<~CSS
.section { color: red; }
.heading:hover { opacity: 0.8; }
CSS
end
def view_template
div(class: "section") do
h2(class: "heading") { "Hello" }
end
end
endCSS is automatically scoped with [data-capsule="..."] attributes and content is wrapped in a scoped element.
ViewComponent
class MyComponent < ApplicationComponent
include StyleCapsule::ViewComponent
def component_styles
<<~CSS
.section { color: red; }
CSS
end
def call
content_tag :div, class: "section" do
"Hello"
end
end
endERB Templates
<%= style_capsule do %>
<style>
.section { color: red; }
</style>
<div class="section">Content</div>
<% end %>CSS Scoping Strategies
StyleCapsule supports two CSS scoping strategies:
-
Selector Patching (default): Adds
[data-capsule="..."]prefix to each selector- Better browser support (all modern browsers)
- Output:
[data-capsule="abc123"] .section { color: red; }
-
CSS Nesting (optional): Wraps entire CSS in
[data-capsule="..."] { ... }- More performant (no CSS parsing needed)
- Requires CSS nesting support (Chrome 112+, Firefox 117+, Safari 16.5+)
- Output:
[data-capsule="abc123"] { .section { color: red; } }
Configuration
Per-component:
class MyComponent < ApplicationComponent
include StyleCapsule::Component
css_scoping_strategy :nesting # Use CSS nesting
endGlobal (in base component class):
class ApplicationComponent < Phlex::HTML
include StyleCapsule::Component
css_scoping_strategy :nesting # Enable for all components
endNote: If you change the strategy and it doesn't take effect, clear the CSS cache:
MyComponent.clear_css_cacheStylesheet Registry
For better performance, register styles for head rendering instead of rendering <style> tags in the body:
class MyComponent < ApplicationComponent
include StyleCapsule::Component
stylesheet_registry namespace: :admin # Optional namespace
def component_styles
<<~CSS
.section { color: red; }
CSS
end
endWith cache strategy:
class MyComponent < ApplicationComponent
include StyleCapsule::Component
stylesheet_registry namespace: :admin, cache_strategy: :time, cache_ttl: 1.hour
def component_styles
<<~CSS
.section { color: red; }
CSS
end
endThen in your layout:
<head>
<%= stylesheet_registry_tags %>
<%= stylesheet_registry_tags(namespace: :admin) %>
</head>Or in Phlex (requires including StyleCapsule::PhlexHelper):
head do
stylesheet_registry_tags
endRegistering Stylesheet Files
You can also register external stylesheet files (not inline CSS) for head rendering:
In ERB:
<% register_stylesheet("stylesheets/user/order_select_component", "data-turbo-track": "reload") %>
<% register_stylesheet("stylesheets/admin/dashboard", namespace: :admin) %>In Phlex (requires including StyleCapsule::PhlexHelper):
def view_template
register_stylesheet("stylesheets/user/order_select_component", "data-turbo-track": "reload")
register_stylesheet("stylesheets/admin/dashboard", namespace: :admin)
div { "Content" }
endIn ViewComponent (requires including StyleCapsule::ViewComponentHelper):
def call
register_stylesheet("stylesheets/user/order_select_component", "data-turbo-track": "reload")
register_stylesheet("stylesheets/admin/dashboard", namespace: :admin)
content_tag(:div, "Content")
endRegistered files are rendered via stylesheet_registry_tags in your layout, just like inline CSS.
Caching Strategies
No Caching (Default)
class MyComponent < ApplicationComponent
include StyleCapsule::Component
stylesheet_registry # No cache strategy set (default: :none)
endTime-Based Caching
stylesheet_registry cache_strategy: :time, cache_ttl: 1.hour # Using ActiveSupport::Duration
# Or using integer seconds:
stylesheet_registry cache_strategy: :time, cache_ttl: 3600 # Cache for 1 hourCustom Proc Caching
stylesheet_registry cache_strategy: ->(css, capsule_id, namespace) {
cache_key = "css_#{capsule_id}_#{namespace}"
should_cache = css.length > 100
expires_at = Time.now + 1800
[cache_key, should_cache, expires_at]
}Note: cache_strategy accepts Symbol (:time), String ("time"), or Proc. Strings are automatically converted to symbols.
File-Based Caching (HTTP Caching)
Writes CSS to files for HTTP caching. Requires class method def self.component_styles:
class MyComponent < ApplicationComponent
include StyleCapsule::Component
stylesheet_registry cache_strategy: :file
# Must use class method for file caching
def self.component_styles
<<~CSS
.section { color: red; }
CSS
end
endConfiguration:
# config/initializers/style_capsule.rb
StyleCapsule::CssFileWriter.configure(
output_dir: Rails.root.join("app/assets/builds/capsules"),
filename_pattern: ->(component_class, capsule_id) {
"capsule-#{capsule_id}.css"
}
)Precompilation:
bin/rails style_capsule:build # Build CSS files
bin/rails style_capsule:clear # Clear generated filesFiles are automatically built during bin/rails assets:precompile.
Compatibility: The stylesheet registry works with Propshaft, Sprockets, and other Rails asset bundlers. Static file paths are collected in a process-wide manifest (similar to Propshaft's approach), while inline CSS is stored per-request.
Advanced Usage
Database-Stored CSS
For CSS stored in a database (e.g., user-generated styles, themes), use StyleCapsule's CSS processor directly:
# app/models/theme.rb
class Theme < ApplicationRecord
def generate_capsule_id
return capsule_id if capsule_id.present?
scope_key = "theme_#{id}_#{name}"
self.capsule_id = "a#{Digest::SHA1.hexdigest(scope_key)}"[0, 8]
save! if persisted?
capsule_id
end
def scoped_css
return scoped_css_cache if scoped_css_cache.present? &&
scoped_css_updated_at == updated_at
current_capsule_id = generate_capsule_id
scoped = StyleCapsule::CssProcessor.scope_selectors(css_content, current_capsule_id)
update_columns(
scoped_css_cache: scoped,
scoped_css_updated_at: updated_at,
capsule_id: current_capsule_id
)
scoped
end
endUsage:
<div data-capsule="<%= theme.capsule_id %>">
<style><%= raw theme.scoped_css %></style>
<div class="header">Content</div>
</div>CSS Selector Support
- Regular selectors:
.section,#header,div.container - Pseudo-classes and pseudo-elements:
.button:hover,.item::before - Multiple selectors:
.a, .b, .c { color: red; } - Component-scoped selectors:
:host,:host(.active),:host-context(.theme-dark) - Media queries:
@media (max-width: 768px) { ... }
Requirements
- Ruby >= 3.0
- Rails >= 6.0, < 9.0 (optional, for Rails integration)
- ActiveSupport >= 6.0, < 9.0 (optional, for Rails integration)
Note: The gem can be used without Rails! See Non-Rails Support below.
Non-Rails Support
StyleCapsule can be used without Rails! The core functionality is framework-agnostic.
Standalone Usage
require 'style_capsule'
# Direct CSS processing
css = ".section { color: red; }"
capsule_id = "abc123"
scoped = StyleCapsule::CssProcessor.scope_selectors(css, capsule_id)
# => "[data-capsule=\"abc123\"] .section { color: red; }"Phlex Without Rails
require 'phlex'
require 'style_capsule'
class MyComponent < Phlex::HTML
include StyleCapsule::Component
def component_styles
<<~CSS
.section { color: red; }
CSS
end
def view_template
div(class: "section") { "Hello" }
end
endSinatra
require 'sinatra'
require 'style_capsule'
class MyApp < Sinatra::Base
helpers StyleCapsule::StandaloneHelper
get '/' do
erb :index
end
end<!-- views/index.erb -->
<%= style_capsule do %>
<style>
.section { color: red; }
</style>
<div class="section">Content</div>
<% end %>Stylesheet Registry Without Rails
The stylesheet registry automatically uses thread-local storage when ActiveSupport is not available:
require 'style_capsule'
# Works without Rails
StyleCapsule::StylesheetRegistry.register_inline(".test { color: red; }", namespace: :test)
stylesheets = StyleCapsule::StylesheetRegistry.request_inline_stylesheetsFor more details, see docs/non_rails_support.md.
How It Works
- Scope ID Generation: Each component class gets a unique scope ID based on its class name (shared across all instances)
-
CSS Rewriting: CSS selectors are rewritten to include
[data-capsule="..."]attribute selectors - HTML Wrapping: Component content is automatically wrapped in a scoped element
- No Class Renaming: Class names remain unchanged (unlike Shadow DOM)
Development
bundle install
bundle exec appraisal install
# Run tests
bundle exec rspec
# Run tests for all Rails versions
bundle exec appraisal rails72 rspec
bundle exec appraisal rails8ruby34 rspec
# Linting
bundle exec standardrb --fixContributing
Bug reports and pull requests are welcome on GitHub at https://github.com/amkisko/style_capsule.rb
Contribution policy:
- New features are not necessarily added to the gem
- Pull requests should have test coverage and changelog entry
Review policy:
- Critical fixes: up to 2 calendar weeks
- Pull requests: up to 6 calendar months
- Issues: up to 1 calendar year
Publishing
rm style_capsule-*.gem
gem build style_capsule.gemspec
gem push style_capsule-*.gemSecurity
StyleCapsule includes security protections:
- Path traversal protection
- Input validation
- Size limits (1MB per component)
- XSS prevention via Rails' HTML escaping
For detailed security information, see SECURITY.md.
License
The gem is available as open source under the terms of the MIT License.