---
name: hotwire-coder
description: Use when implementing Hotwire features with Turbo Drive, Turbo Frames, and Turbo Streams. Applies Rails 8 conventions, morphing, broadcasts, lazy loading, and real-time update patterns.
allowed-tools: Read Write Edit Grep Glob Bash WebSearch
---
# Hotwire Coder
## Hotwire Philosophy
Hotwire sends HTML over the wire instead of JSON:
- **Turbo Drive** - Accelerates navigation by replacing `
` without full page reloads
- **Turbo Frames** - Decompose pages into independent contexts that update on request
- **Turbo Streams** - Deliver partial page updates from server (HTTP or WebSocket)
## Turbo Drive
### Selective Opt-Out
```erb
<%= link_to "Download PDF", report_path(format: :pdf), data: { turbo: false } %>
<%= form_with model: @legacy, data: { turbo: false } do |f| %>
```
### Form Submissions
```ruby
def create
@post = Post.new(post_params)
@post.save ? redirect_to(@post, notice: "Created!") : render(:new, status: :unprocessable_entity)
end
```
## Turbo Frames
### Basic Frame
```erb
```
### Lazy Loading
```erb
Loading...
```
See `references/lazy-loading.md` for skeleton UI patterns, infinite scroll, and Stimulus loading controllers.
### Breaking Out of Frames
```erb
<%= link_to "View Post", post_path(@post), data: { turbo_frame: "_top" } %>
<%= link_to "Results", search_path, data: { turbo_frame: "results" } %>
```
### Inline Editing Pattern
```erb
<%= post.title %>
<%= link_to "Edit", edit_post_path(post) %>
```
## Turbo Streams
### Stream Actions
```erb
<%= turbo_stream.append "comments" do %><%= render @comment %><% end %>
<%= turbo_stream.replace dom_id(@post) do %><%= render @post %><% end %>
<%= turbo_stream.remove dom_id(@comment) %>
```
### HTTP Stream Responses
```ruby
def create
@comment = @post.comments.create(comment_params)
respond_to do |format|
format.turbo_stream # renders create.turbo_stream.erb
format.html { redirect_to @post }
end
end
```
### Real-Time Broadcasts
```ruby
class Comment < ApplicationRecord
after_create_commit -> {
broadcast_append_to post, target: "comments", partial: "comments/comment"
}
after_update_commit -> { broadcast_replace_to post }
after_destroy_commit -> { broadcast_remove_to post }
end
```
```erb
<%= turbo_stream_from @post %>
```
> **Note:** For LLM streaming or features requiring message delivery guarantees, see `anycable-coder` skill. Action Cable provides at-most-once delivery which can lose chunks on reconnection.
## Turbo 8 Morphing
```erb
```
```ruby
class Post < ApplicationRecord
after_update_commit -> { broadcast_refresh_to self }
end
```
## Common Patterns
### Modal Pattern
```erb
<%= link_to "New Post", new_post_path, data: { turbo_frame: "modal" } %>
```
### Infinite Scroll
```erb
<%= render @posts %>
<% if @posts.next_page? %>
<% end %>
```
## Anti-Patterns
| Anti-Pattern | Problem | Solution |
|--------------|---------|----------|
| Mismatched frame IDs | Silent failures | Validate IDs match |
| Missing status codes | Turbo ignores response | Use 422/303 correctly |
| Implicit locals in broadcasts | Runtime errors | Always pass `request_id: nil` |
## Debugging
```javascript
Turbo.setDebug(true)
document.addEventListener("turbo:frame-missing", (e) => console.error("Frame not found:", e.detail.response))
```
## Output Format
When implementing Hotwire features, provide:
1. **Controller** - Actions with proper response handling
2. **Views** - HTML/ERB with frames and streams
3. **Model** - Broadcast callbacks if real-time needed
4. **JavaScript** - Stimulus controllers if needed