The All-React App
Look at what the Stimulus page looks like in React in this lesson.
We'll cover the following...
In creating the React page, we made a few starting decisions:
- We didn’t need styled components, because the CSS already existed.
- We did not try to make a React route bridge between the two React pages. Instead, the schedule page still uses the Rails server to route between the two pages.
- There’s some minor date parsing and formatting on this page. Rather than introduce a new JavaScript library to manage this, we did it on the server-side and had the formatted and parsed data made part of the data sent to React.
- We decided to keep the show/hide behavior via CSS classes, even though React makes it relatively easy to just remove and reinsert the DOM elements. This was to keep from also having to rewrite the tests and to avoid more complicated logic in the React components.
We also decided that we couldn’t quite use the existing filter reducer (app/javascript/src/calendar_filter_store.ts
) as is, although we did adapt the logic into a Redux reducer.
The only other real design decision was how to break up the components. We made individual calendar days at the top components, individual schedule days, and individual concerts, with some grouping components to manage the lists.
We have to roll the markup up into React components and then make those components talk to and receive data from the reducers. Let’s start on the Rails side.
The controller doesn’t actually have to change, but we do reduce the view file to just this:
<div id="schedule-react-element"data-schedule="<%= @schedule.days_hash.to_json %>"data-sold-out-concert-ids="<%= @concerts.select(&:sold_out?).map(&:id).to_json %>"></div>
Very similar to the other React app, we’re creating a div
element with a known ID that we’ll use to attach the React app, and we’re passing it elements that will become React props. These props are a list of the IDs of sold out concerts and the schedule data as a hash.
The schedule.to_hash
method is new. Here’s the whole file:
#---# Excerpted from "Modern Front-End Development for Rails",# published by The Pragmatic Bookshelf.# Copyrights apply to this code. It may not be used to create training material,# courses, books, articles, and the like. Contact us if you are in doubt.# We make no guarantees that this code is fit for any purpose.# Visit http://www.pragmaticprogrammer.com/titles/nrclient for more book information.#---class Scheduleattr_accessor :schedule_daysdef self.from_concerts(all_concerts)schedule = Schedule.newall_concerts.group_by(&:start_day).each do |day, concerts|schedule.schedule_days << ScheduleDay.new(day, concerts)schedule.schedule_days.sort_by!(&:day)endscheduleenddef initialize@schedule_days = []enddef days_hashschedule_days.sort_by(&:day).map(&:to_h)endend
A schedule is a dictionary in which the keys are the days in the schedule, and the values are a list of concerts. The ScheduleDay
class also grows to a hash method:
#---# Excerpted from "Modern Front-End Development for Rails",# published by The Pragmatic Bookshelf.# Copyrights apply to this code. It may not be used to create training material,# courses, books, articles, and the like. Contact us if you are in doubt.# We make no guarantees that this code is fit for any purpose.# Visit http://www.pragmaticprogrammer.com/titles/nrclient for more book information.#---class ScheduleDayattr_accessor :day, :concertsdef initialize(day, concerts)@day = day@concerts = concertsenddef day_to_h{month: day.by_example("Jan"),date: day.by_example("02"),year: day.by_example("2006"),weekday: day.by_example("Monday"),key: day.by_example("2006-01-02")}enddef to_h{id: day.by_example("2006-01-02"),day: day_to_h,concerts: concerts.sort_by(&:start_time).map(&:to_h)}endend
Here you see some of the data massaging we’ve decided to do server-side rather than deal with it client-side. In particular, the ...