This repository contains code and documentation for Jesse Spevack's Railsconf 2021 talk "Stimulating Events".
We are starting with a brand new rails application that I've added Tailwind css to. We'd like to use Stimulus to add a really nice professional, modern sheen to a simple dropdown menu.
The first thing we are going to do is generate a new demo controller with a dropdown method.
rails generate controller Demo dropdown
Let's open our dropdown.html.erb view in the view/demo directory. I'm going to paste some html for our dropdown that I've put together with the help of Tailwind UI, which is a premium html component library by the folks who made tailwind css.
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
<div class="bg-white overflow-hidden shadow rounded-lg divide-y divide-gray-200">
<div class="px-4 py-5 sm:px-6">
<h1 class="text-lg leading-6 font-bold text-gray-900">Stimulating Events</h1>
<h2 class="text-lg leading-6 font-medium text-gray-900">Railsconf 2021</h2>
<p class="mt-1 text-sm text-gray-500">
Jesse Spevack
</p>
</div>
<div class="px-4 py-5 sm:p-6 mb-48">
<label id="listbox-label" class="block text-sm font-medium text-gray-700">
Railsconf Keynote
</label>
<div class="mt-1 relative">
<button type="button" aria-haspopup="listbox" aria-expanded="true" aria-labelledby="listbox-label" class="bg-white relative w-full border border-gray-300 rounded-md shadow-sm pl-3 pr-10 py-2 text-left cursor-default focus:outline-none focus:ring-1 focus:ring-pink-500 focus:border-pink-500 sm:text-sm">
<span class="block truncate">
Select speaker
</span>
<span class="absolute inset-y-0 right-0 flex items-center pr-2 pointer-events-none">
<svg class="h-5 w-5 text-gray-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
<path fill-rule="evenodd" d="M10 3a1 1 0 01.707.293l3 3a1 1 0 01-1.414 1.414L10 5.414 7.707 7.707a1 1 0 01-1.414-1.414l3-3A1 1 0 0110 3zm-3.707 9.293a1 1 0 011.414 0L10 14.586l2.293-2.293a1 1 0 011.414 1.414l-3 3a1 1 0 01-1.414 0l-3-3a1 1 0 010-1.414z" clip-rule="evenodd" />
</svg>
</span>
</button>
<div class="absolute mt-1 w-full rounded-md bg-white shadow-lg">
<ul tabindex="-1" role="listbox" aria-labelledby="listbox-label" aria-activedescendant="listbox-item-3" class="max-h-60 rounded-md py-1 text-base ring-1 ring-black ring-opacity-5 overflow-auto focus:outline-none sm:text-sm">
<% @speakers.each do |speaker| %>
<li role="option" class="text-gray-900 cursor-default select-none relative py-2 pl-3 pr-9">
<span class="font-normal block truncate">
<%= speaker %>
</span>
<span class="text-pink-600 absolute inset-y-0 right-0 flex items-center pr-4">
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
<path fill-rule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clip-rule="evenodd" />
</svg>
</span>
</li>
<% end %>
</ul>
</div>
</div>
</div>
</div>
</div>
Now lets open the demo controller we just generated in the app controllers directory.
We are going to be building a dropdown menu where a user can choose a Railsconf keynote speaker. We'll add some data to our dropdown method in our controller. Typically in a real application this might be accomplished via database query or some other service request.
# app/controllers/demo_controller.rb
class DemoController < ApplicationController
def dropdown
@keynote_speakers = [
'David Heinemeier Hannson',
'Eileen M. Uchitelle',
'Aaron Patterson',
'Bryan Cantrill'
]
end
end
And we can view this page now by opening our browser.
localhost:3000/demo/dropdown
Now lets use Stimulus to get this drop down to work. We'll start by copying the hello_controller.js
to get the Stimulus boilerplate for our dropdown. We'll call our new stimulus controller dropdown_controller.
cp app/javascript/controllers/hello_controller.js app/javascript/controllers/dropdown_controller.js
Now it is time to link our html view to our new dropdown controller. We do this with an identifier, which is the name we use to reference a controller class in our html. Our dropdown has two parts. We are using a button tag as the selector. We are using an unordered list as the options the user can select. Let's find the div
that wraps the button and the unordered list. We'll add a data controller attribute to this div
, which will give the dropdown controller scope that includes the wrapper div
as well as both our button and our unordered list.
<div data-controller="dropdown">
<button>...</button>
<ul>...</ul>
</div>
Let's add a console.log
in the dropdown_controller.js
to show that our controller is properly connected to our html. We can use one of the Stimulus lifecylce methods, connect
, which is called anytime the controller is connected to the DOM.
connect() {
console.log("Hello, Stimulus!")
}
To achieve this, we have some Tailwind css classes that define styles we'd like to apply to elements on our page using the class attribute. Stimulus lets us refer to CSS classes by logical naming using a combination of data attributes and controller properties. We type data-<the controller name>-class="<the css class to apply>"
.
This makes a Stimulus controller re-useable as the styles we apply can be dynamically passed in to our controller on a page by page basis. In this case we want to add some white, pink, and bold styles. But we can imagine using different styles on another dropdown in the same application.
<div
data-controller="dropdown"
data-dropdown-text-white-class="text-white"
data-dropdown-text-pink-class="text-pink-600"
data-dropdown-background-pink-class="bg-pink-600"
data-dropdown-semibold-class="font-semibold"
data-dropdown-hidden-class="hidden"
class="mt-1 relative"
>
And we also need to add each of these classes to our controller's static classes
array:
// app/javascript/controllers/dropdown_controller.js
static classes = [
"textWhite",
"textPink",
"backgroundPink",
"semibold",
"hidden"
]
So adding
<li data-action="mouseenter->dropdown#highlightListItem">...</li>
We have to define our highlightListItem
function, so we can add some console logging to make sure we've got it right. Every action sends the triggering event to the function in our specified Stimulus controller. This event object includes a target
, which is the target that dispatched the event. In this case the list item. So let's console log the inner text of the first child element of the list item. This should print out the name of the speaker mouse'ed over in the console.
// app/javascript/controllers/dropdown_controller.js
highlightListItem(event) {
console.log("Mouse enter: ", event.target.innerText)
}
Since we see this message as we cursor over our list of speakers, lets now use javascript to apply our desired style.
First, We want the list item's background to turn pink. So lets start by adding this.backgroundPinkClass, which points to the class we defined in our static classes array, to the event target, the element that dispatched the mouse enter event, which is the list item.
This looks like it's working, so now let's od the same thing for font color, which we'll want to turn white, and the same thing for the text which we'll want to turn bold. The text that we want to turn white and bold is actually in a span nested within the list item. So we'll call first element child on the list item to get the span we are after.
// app/javascript/controllers/dropdown_controller.js
highlightListItem(event) {
const listItem = event.target
const span = listItem.firstElementChild
listItem.classList.add(this.backgroundPinkClass)
span.classList.add(this.textWhiteClass, this.semiboldClass)
}
Now as we move our mosue over our list, each list item will get a pink background and the inner span text will turn white and bold. Almost there. But we also want to unhighlight the list item when our mouse leaves that particular list item. To do so, we need to define another action on our list item.
<li data-action="mouseleave->dropdown#unhighlightListItem mouseenter->dropdown#highlightListItem">
...
</li>
Here the DOM event is mouseleave
so when the mouse leaves the list item, the unhighlightListItem
function in the dropdown
controller will get called.
Let's implement that function.
// app/javascript/controllers/dropdown_controller.js
unhighlightListItem(event) {
const listItem = event.target
const span = listItem.firstElementChild
listItem.classList.remove(this.backgroundPinkClass)
span.classList.remove(this.textWhiteClass, this.semiboldClass)
}
At this point our list items are highlighted and unhighlighted as our mouse enters or leaves them.
Now we need to handle the actual selection of a list item. We'll add another action. This time the DOM event we care about is a click
event. We'll want the click
event on a list item to trigger a new function selectItem
that we will define in our dropdown
controller.
<li data-action="click->dropdown#selectItem mouseleave->dropdown#unhighlightListItem mouseenter->dropdown#highlightListItem">
...
</li>
And now we can add a console log within our new selectItem
to make sure things are wired up correctly.
selectItem(event) {
console.log("Selecting: ", event.target.innerText)
}
Our console shows that we are logging the correct selected list item. Now we will want to do two things. First, need to hide the list since the selection has been made. Second, we'll want to add the user's selection to the inner text of the button, so they see that their selection is made.
To do this we need an easy way to get to the text inside our button. For that, Stimulus gives us the concept of targets, which allow us to reference important elements by name in our controller.
We'll add a data-<controller>-target="name of the target"
in our html.
<span data-dropdown-target="speaker">
Select speaker
</span>
And then in our controller we can add the speaker
target to our static targets
array.
static targets = [
"speaker"
]
This will allow us to reference this span element with the text we want to update in our selectItem
function. We do so by calling this
and then the name of the target followed by Target
. This will give us the element that corresponds with the data-dropdown-target
equal to speaker
.
selectItem(event) {
this.speakerTarget.textContent = event.target.innerText
}
Now when we click on a list item, the text of the button will update to the text of the list item that was clicked.
I'm just going to clean up this function a bit.
The second thing we wanted to do was hide the drop down list after a selection gets made. To do so we'll add another target in our html to help us more easily reference elements from our html in our controller. Let's add another data-dropdown-target
with the value list
to our unordered list tag.
<ul data-dropdown-target="list">
...
</ul>
And then in our controller we can add the list
target to our static targets
array.
static targets = [
"speaker",
"list"
]
And then we can hide this list in the selectItem
by adding the hidden
class to it.
selectItem(event) {
this.selectTarget.textContent = event.target.innerText
this.listTarget.classList.add(this.hiddenClass)
}
Now we can select away our list, but there is no way to get it back. We can fix this by adding a click action to our button so that when the user clicks the button the list will reappear. Stimulus gives certain special elements, like buttons, their own default event short hands. So because this is a click event on a button element, we do not need to specify click
we only need the data-action
attribute to include the controller name and the function to call. So in this case rather than writing click arrow dropdown # toggleList we can just write dropdown # toggleList and the click is inferred because this is a button element.
<button data-action="dropdown#toggleList">
...
</button>
And then our toggleList
function can toggle the hidden class on our list target.
toggleList() {
this.listTarget.classList.toggle(this.hiddenClass)
}
And because this functionality is really similar to what we just put in the selectItem
function we can refactor and DRY up our implementation.
selectItem(event) {
this.selectTarget.textContent = event.target.innerText
this.toggleList()
}
We can also add this to our connect
function so that the drop down is not visible on page load.
connect() {
this.toggleList()
}
At this point our list mostly works. It would be great to get those check marks to show up nicely when we highlight. To do this we are going to define a new function called hideChecks
that will iterate over our list items and uncheck any that do not have our selected value.
We will first add a new data-dropdown-target
target called check
that will reference the span containing our check svg icon.
<span data-dropdown-target="check">
<svg></svg>
</span>
Next we'll add this target to our static targets
array.
static targets = [
"speaker",
"list",
"check"
]
Now we can implement our hideChecks
function. In the past we've used singular targets, but our page actually has multiple checks, one per list item. So we can use the plural targets suffix to grab an array of our check targets. We'll loop over each of them and apply our hidden class. Let's make sure we are looping correctly by console logging the check element.
hideChecks() {
for (const check of this.checkTargets) {
check.classList.add(this.hiddenClass)
}
}
We can add this function to our connect
class so that none of the speakers is checked on page load.
connect() {
this.toggleList()
this.hideChecks()
}
Now when the user selects an item and re-opens the dropdown we want the selected item to have a visible check icon.
selectItem(event) {
this.toggleList()
this.hideChecks()
const selected = event.target.innerText.trim()
this.selectTarget.textContent = selected
for(const check of this.checkTargets) {
if(check.previousElementSibling.innerText.trim() === selected) {
check.classList.remove(this.hiddenClass)
}
}
}
And because we love our work, we will turn the check marks white when their parent list item is highlighted. To do so we'll add to the highlightListItem
and unhighlightListItem
functions.
highlightListItem(event) {
const listItem = event.target
const span = listItem.firstElementChild
const check = listItem.lastElementChild
listItem.classList.add(this.backgroundPinkClass)
span.classList.add(this.textWhiteClass, this.semiboldClass)
if (!check.classList.contains(this.hiddenClass)) {
check.classList.replace(this.textPinkClass, this.textWhiteClass)
}
}
And we will do the reverse in unhighlight.
unhighlightListItem(event) {
const listItem = event.target
const span = listItem.firstElementChild
const check = listItem.lastElementChild
listItem.classList.remove(this.backgroundPinkClass)
span.classList.remove(this.textWhiteClass, this.semiboldClass)
if (!check.classList.contains(this.hiddenClass)) {
check.classList.replace(this.textWhiteClass, this.textPinkClass)
}
}
At this point we have some classes that toggle on and off. We could imagine adding a post method when we select an item that stores the selection in our database, but that is beyond the scope of this talk.
Instead, lets look at our code. Does it bring us joy? It's kind of confusing if I'm being honest. There are a lot of things to keep track of. We loop over checks in a few places. We might be able to DRY the code out a bit, but I think this is a sign that maybe our dropdown controller is not well encapsulated.
Let's break our dropdown controller into two chunks. We'll keep the logic for opening and closing our list and setting the selected text in the dropdown controller. If we were posting the selected data to one of our Rails controllers, we might do that here as well, or potentially in a third controller altogether.
We'll also add a new list_item_controller
and empower each list item to manage it's own highlight styling.
First we'll make a new controller with the necessary Stimulus boilerplate.
cp app/javascript/controllers/hello_controller.js app/javascript/controllers/list_item_controller.js
Now around each list item we'll add a div
on which to instantiate our new controller. That way the list item controller will have a single list item in its scope. We'll type data-controller = list item.
<div data-controller="list-item">
<li></li>
</div>
Let's hook send a message to our console to make sure everything is hooked up correctly.
connect() {
console.log("Hello, Stimulus!")
}
Now let's add some targets in our new list item controller so that it is easy to reference the important elements in our list item. First the entire list item, which we'll call item. Next we need the check, and finally, let's also add a data list item target equals text to make it easy to find the name of the speaker.
<li data-list-item-target="item" ...>
<span data-list-item="text" ...>
<%= speaker %>
</span>
<span data-list-item="check" ...>
<svg></svg>
</span>
</li>
We need to add these targets to our static targets
array.
static targets = [
"text", // the span containing the speaker's name
"check", // the span containing the check svg icon
"item" // the entire list item
]
Now lets move the data-dropdown
class attributes to our list item. We will add these to the the div
where we attached our listItem
controller.
<div
data-controller="list-item"
data-list-item-text-white-class="text-white"
data-list-item-text-pink-class="text-pink-600"
data-list-item-background-pink-class="bg-pink-600"
data-list-item-semibold-class="font-semibold"
data-list-item-normal-class="font-normal"
>
<li></li>
</div>
And we also need to add these classes to the static classes
array. We can delete the unneeded css classes from our dropdown controller. I think all we'll need is the hidden class to toggle the dropdown list.
static classes = [
"textWhite",
"textPink",
"backgroundPink",
"semibold",
"normal"
]
And we also have to re-write our data actions defined on our list items We can shorten this to a select, highlight and unhighlight method. The list item is redundant because of our new controller.
<li data-action="click->list-item#select mouseenter->list-item#highlight mouseleave->list-item#unhighlight">
</li>
Now let's make sure our actions are firing.
select(event) {
console.log("Selecting: ", event.target.innerText.trim())
}
highlight(event) {
console.log("Highlighting: ", event.target.innerText.trim())
}
unhighlight(event) {
console.log("UnHighlighting: ", event.target.innerText.trim())
}
Now let's move our highlighting functionality over. Because we defined the elements we care about as targets, we can reference them more directly instead of needing to get at them by way of first or last element child.
First will add a pink background to our list item.
Next we'll add the white bold font to our span text.
And finally we'll apply the same white color to our check.
highlight(event) {
this.itemTarget.classList.add(this.backgroundPinkClass)
this.textTarget.classList.add(this.textWhiteClass, this.semiboldClass)
this.checkTarget.classList.replace(this.textPinkClass, this.textWhiteClass)
}
And this reads a lot nicer - do this to the item, and this to the text, and this to the check.
We can do the opposite for our unhighlight
unhighlight(event) {
this.itemTarget.classList.remove(this.backgroundPinkClass)
this.textTarget.classList.remove(this.textWhiteClass, this.semiboldClass)
this.checkTarget.classList.replace(this.textWhiteClass, this.textPinkClass)
}
For select, we are going to want to reach for another tool in the Stimulus tool box. We are going to use data values to store the state of our list items. We'll first add a new attribute to our div
with our list item controller that will start with data
hyphen the controller hyphen the name of the value hyphen value
and set that equal to whatever value we want to store.
<div ... data-list-item-is-selected-value=false>
We can store booleans, integers, strings among other types of data here. In this case we want a boolean value that tells us whether or not the list item is in a selected state.
We will add this value to our static values object in the controller.
static values = {
isSelected: Boolean
}
Now when we connect, we can log this value. And our log shows the value is false, which is what we put in the html data attribute
connect() {
console.log("isSelected: ", this.isSelectedValue)
}
And now our select function becomes a little cleaner. Instead of messing with the list item text, we'll just set the isSelected
state to be true.
select() {
this.isSelectedValue = true
}
These values in Stimulus also come with other powers. For example, we can use a callback function that gets triggered whenever this value gets changed. These functions are name by the name of the value followed by value followed by the word Changed
.
isSelectedValueChanged() {
console.log("isSelected value changed to: ", this.isSelectedValue)
}
When a list item is selected, we want to show the check mark, so we'll remove the hidden class.
isSelectedValueChanged() {
this.checkTarget.classList.toggle(this.hiddenClass)
}
By default we see that the changed callback is triggered when the controller is first instantiated, so all of the checks will get hidden initially.
Now our list item styles work, but we need to somehow create a line of communication between our new list item controller and our old dropdown controller and the sibling list item controllers. When a particular list item is selected, the controller associated with that item should announce "Hey, I've been selected! I'm the one!" and the sibling list items should all uncheck themselves and the dropdown should hide the list.
To do this we'll create a custom event and dispatch it when a list item is selected. We'll create a detail hash that will contain the text of the list item that is being selected. We'll call this custom event the listItemSelected
event.
select() {
this.selectedValue = true
const detail = { value: this.textTarget.innerText.trim() }
const event = new CustomEvent('listItemSelected', { detail: detail }
window.dispatchEvent(event)
}
In our html we'll add data actions to listen for this new custom event. First on our list item controller div
we'll want to listen for this event and then de-select the list items sibling list items.
<div ... data-action="listItemSelected@window->list-item#deselect">
<li></li>
</div>
And then we can define the deselect function to with a console log to make sure things are working. We will log the value of the selected item that dispatched the event and the value of the item that just heard the event. We'll also get rid of the event getting passed into select, because three lines later we define a new event and that will cause us problems.
deselect(event) {
console.log("The selected item's value is: ", event.detail.value)
console.log("This item's value is: ", this.textTarget.innerText.trim())
}
So that appears to be working.
When these two values are the same, we are dealing with the list item that published the listItemSelected
event. So we can just ignore it. But on the other sibling list items, we need to set the isSelected
state to false
.
deselect(event) {
if(event.detail.value !== this.textTarget.innerText.trim()) {
this.isSelectedValue = false
}
}
This will trigger the isSelectedValueChanged
function, which will uncheck our list item appropriately.
At this point our list items are more or less autonomous. They are much more encapsulated than in our initial implementation.
Now all that is left to do is clean up our drop down controller.
First we need to remove the functions we no longer need. selectItem
, setCheck
, hideCheck
, highlightListItem
and unhighlightListItem
. These are all either no longer needed or handled by our list item controller.
Now we'll want to add a new function to set the selected value. We can add a new data action to our button
that will listen for any listItemSelected
event and then invoke the a new set selected function in our dropdown controller.
<div ... data-action="listItemSelected@window->dropdown#setSelected">
...
</div>
And that function just has to change the speaker
target textContent if the event contains a value.
setSelected(event) {
if (event.detail.value) {
this.speakerTarget.textContent = event.detail.value
}
}
And then in the connect
function we'll want to make sure we are still toggling the list off when the page loads.
connect() {
this.toggleList()
}
Thanks for sticking with me through this. We covered a lot. We looked at how to create a controller, use actions, targets, values and css classes. We also used custom events to communicate between our controllers and better encapsulate our code. I hope you are walking away with an increased interest in Stimulus and maybe a few new tools in your stimulus repertoire. I'll be answering any questions in Discord.
Thank you again for joining me! Enjoy the rest of Railsconf.