In this step-by-step guide, I will show you how to create a Twitter clone in pure Ruby with Matestack, following the great screencasts from Chris McCord Phoenix LiveView Twitter Clone and Nate Hopkins Stimulus Reflex Twitter Clone. We will use the Gem matestack-ui-core, which enables us to implement our UI in some Ruby classes rather than writing ERB, HAML or Slim views. Furthermore we don't need to touch JavaScript in order to create reactive UI features, such as updating the DOM without a full browser page reload or syncing multiple web clients through Action Cable!
I've added a screencast (two parts) showing you what you will be doing in this tutorial:
This guide utilizes the full power of Matestack and uses matestack-ui-core as a complete substitute for Rails views. If you only want to create UI components in pure Ruby on existing Rails views, please check outthis guide
// This file is automatically compiled by Webpack, along with any other files// present in this directory. You're encouraged to place your actual application logic in// a relevant structure within app/javascript and only use these pack files to reference// that code so it'll be compiled.require("@rails/ujs").start()// require("turbolinks").start() //removerequire("@rails/activestorage").start()require("channels")import MatestackUiCore from'matestack-ui-core'//add
classTwitterClone::App<Matestack::Ui::Appdefresponse div class:"container"do heading size:1,text:"Twitter Clone",class:"mb-5" yield_pageendendend
app/matestack/twitter_clone/pages/posts/index.rb
classTwitterClone::Pages::Posts::Index<Matestack::Ui::Pagedefprepare @posts =Post.allenddefresponse @posts.eachdo|post| div class:"mb-3 p-3 rounded shadow-sm"do heading size:5do plain post.username small text: post.created_at.strftime("%d.%m.%Y %H:%M")end paragraph text: post.bodyendendendend
Add Matestack to the controller
app/controllers/posts_controller.rb
classPostsController<ApplicationController matestack_app TwitterClone::App# add this# GET /posts# GET /posts.jsondefindex# @posts = Post.all render TwitterClone::Pages::Posts::Index# add thisend# ...end
Test the current state
railss
You should see the heading "Twitte Clone" and that's it. We don't have any posts in our database, so we need a form to create one!
Add a reactive form
app/matestack/twitter_clone/pages/posts/index.rb
classTwitterClone::Pages::Posts::Index<Matestack::Ui::Pagedefprepare @posts =Post.allenddefresponse post_form_partial post_list_partialendprivatedefpost_form_partial div class:"mb-3 p-3 rounded shadow-sm"do heading size:4,text:"New Post",class:"mb-3" form form_config_helper do div class:"mb-3"do form_input key::username,type::text,placeholder:"Username",class:"form-control"end div class:"mb-3"do form_textarea key::body,placeholder:"What's up?",class:"form-control"end div class:"mb-3"do form_submit do button type::submit,class:"btn btn-primary",text:"Post!"endendendendenddefform_config_helper {for:Post.new,path: posts_path,method::post,# optional: in order to map Bootstrap's CSS classes, you can adjust the form error rendering like so:errors: { wrapper: { tag::div,class:'invalid-feedback' },input: { class:'is-invalid' } } }enddefpost_list_partial @posts.eachdo|post| div class:"mb-3 p-3 rounded shadow-sm"do heading size:5do plain post.username small text: post.created_at.strftime("%d.%m.%Y %H:%M")end paragraph text: post.bodyendendendend
app/controllers/posts_controller.rb
classPostsController<ApplicationController# ...# POST /posts# POST /posts.jsondefcreate @post =Post.new(post_params)# respond_to do |format|# if @post.save# format.html { redirect_to @post, notice: 'Post was successfully created.' }# format.json { render :show, status: :created, location: @post }# else# format.html { render :new }# format.json { render json: @post.errors, status: :unprocessable_entity }# end# endif @post.save render json: {message:'Post was successfully created.' },status::createdelse render json: {errors: @post.errors,message:'Post could not be created.' },status::unprocessable_entityendend# ...end
Test the current state
You should see a basic index page with a form on top. When submitting the form without any values, ActiveRecord errors should appear below the inputs without browser page reload. When submitting valid data, the form should reset automatically without browser page reload, but you would have to reload the browser in order to see the new post! That's why we need to add the async component.
Cool! Now you should see the list automatically updating itself after form submission without a browser page reload! And we didn't write any JavaScript for that. Just two lines of simple Ruby code! How cool is that?
Now we need to add some action components in order to "like" the posts.
Enable "likes"
config/routes.rb
Rails.application.routes.drawdo resources :postsdo member do put 'like',to:'posts#like'endendend
app/controller/posts_controller.rb
# ...# PUT /posts/1/likedeflike @post =Post.find params[:id] @post.increment(:likes_count)if @post.save render json: {message:'Post was successfully liked.' },status::createdelse render json: {errors: @post.errors,message:'Post could not be liked.' },status::unprocessable_entityendend# ...
Great! Now we get instant feedback after performing successful or non successful form submissions! And still no line of JavaScript involved! The same approach would work for our actions, but we do not want to have that feedback after performing the actions in this example!
All of the above described reactivity only applies for one client. A second user wouldn't see the new post, unless he reloads his browser page. But of course, we want to sync all connected clients! It's time to integrate ActionCable!
import consumer from"./consumer"consumer.subscriptions.create("MatestackUiCoreChannel", {connected() {// Called when the subscription is ready for use on the server },disconnected() {// Called when the subscription has been terminated by the server },received(data) {// Called when there's incoming data on the websocket for this channelMatestackUiCore.matestackEventHub.$emit(data.event, data) }});
app/channels/matestack_ui_core_channel.rb
classMatestackUiCoreChannel<ApplicationCable::Channeldefsubscribed stream_from "matestack_ui_core"enddefunsubscribed# Any cleanup needed when channel is unsubscribedendend
app/controller/posts_controller.rb
# ...# PUT /posts/1/likedeflike @post =Post.find params[:id] @post.increment(:likes_count)if @post.saveActionCable.server.broadcast('matestack_ui_core', {event:"cable__liked_post_#{@post.id}" }) render json: {message:'Post was successfully liked.' },status::createdelse render json: {errors: @post.errors,message:'Post could not be liked.' },status::unprocessable_entityendend# POST /postsdefcreate @post =Post.new(post_params)if @post.saveActionCable.server.broadcast('matestack_ui_core', {event:'cable__created_post' }) render json: {message:'Post was successfully created.' },status::createdelse render json: {errors: @post.errors,message:'Post could not be created.' },status::unprocessable_entityendend# ...
Wow! We just had to copy and paste a JavaScript snippet once in order to integrate ActionCable, broadcast an event from the controller action and without any more added complexity, we get synced clients, implemented in pure Ruby! Fantastic!
We will take a short break before adding the next cool reactivity feature add refactor a little bit! Matestack encourages you to create a readable and maintainable UI implemetation. Therefore we will move some complexity from the current index page to a self contained Matestack component!
Everything should be the same! We just refactored some code in order to better manage complexity.
Now we will cover the last topic of this guide:
As described before, the async rerenders it's whole body. The async wrapping the whole post list therefore rerenders ALL posts. If our list of posts grows, the performance of the rerendering will decrease. In a lot of usecases, this will not be an issue since the UI is not too big/too complex. So go ahead and use async everywhere you're not rerendering big or complex UIs and enjoy the simplicity of that rerendering approach!
But now imagine, your post list will be too big at some point. We should switch the reactivity approach to a more granular on. Let's use the cable component alongside our already added ActionCable introduction and reuse pretty much all written code!
# ...# POST /postsdefcreate @post =Post.new(post_params)if @post.saveActionCable.server.broadcast('matestack_ui_core', {event:'cable__created_post',data:matestack_component(:post_component,post: @post)# add this line }) render json: {message:'Post was successfully created.' },status::createdelse render json: {errors: @post.errors,message:'Post could not be created.' },status::unprocessable_entityendend# ...
Test the current state
You probably don't realize any difference on the UI, but now ONLY the fresh post will be rendered on the server and pushed to the cable component mounted in the browser. The cable component is configured to prepend (put on top) everything pushed from the server on the cable__created_post event. This reactivity approach is now already much more scalable in a context of big/complex UI rerendering.
The cable component can prepend, append, update and delete elements within its body or replace its whole body with something pushed from the server. We want to use the update feature in order to rerender a specific post when liked:
# ...# PUT /posts/1/likedeflike @post =Post.find params[:id] @post.increment(:likes_count)if @post.saveActionCable.server.broadcast('matestack_ui_core', {# event: "cable__liked_post_#{@post.id}"# no id required in the event name, the cable component will figure out which post# should be updated using the root element ID of the pushed componentevent:"cable__liked_post",# change the event namedata:matestack_component(:post_component,post: @post)# add this line }) render json: {message:'Post was successfully liked.' },status::createdelse render json: {errors: @post.errors,message:'Post could not be liked.' },status::unprocessable_entityendend# POST /postsdefcreate @post =Post.new(post_params)if @post.saveActionCable.server.broadcast('matestack_ui_core', {event:'cable__created_post',data:matestack_component(:post_component,post: @post)# add this line }) render json: {message:'Post was successfully created.' },status::createdelse render json: {errors: @post.errors,message:'Post could not be created.' },status::unprocessable_entityendend# ...
Test the current state
Again: you probably don't realize any difference on the UI, but now ONLY the updated post will be rendered on the server and pushed to the cable component mounted in the browser.
The cable component is configured to updated the component pushed from the server on the cable__liked_post event. The cable component then reads the ID of the root element of the pushed component, looks for that ID within it's body and updates this element with the pushed component.
Now, we're rerendering the list and its elements completely with the cable component. As described, this is an ALTERNATIVE approach to the introduced async component approach. The cable component requires a bit more implementation and brain power but makes our reactivity more scalable. Use the cable component wherever you think async would be too slow at some point!
Ok, let's lazy load the list of posts in order to speed up initial page load when reading the posts from the database and rendering them gets "too slow" at some point. Take a deep breath: We will use async and cable together now!
Relax, it's super simple:
Lazy load the post list with async's defer feature
That was easy, right? The async requested its content right after the page was loaded. We moved the ActiveRecord query out of the prepare method out of following reason: When rendering a Matestack page/component, the prepare method is always called. This means, the ActiveRecord query is performed on the initial page load although we don't need the data yet. Matestacks rendering mechanism stops rendering components which are wrapped in an async defer component on initial page load and only renders them, when they are explicitly requested in a subsequent HTTP call. Therefore we should take care of calling the ActiveRecord query only from within the deferred block. In our example we accomplish this by calling the helper method posts instead of using the instance variable @posts, formerly resolved in the prepare method.
Using this approach, it is super simple to speed up initial page loads without adding complexity or JavaScript to your code! Awesome!
Want some sugar? How about adding a CSS animation while lazy loading the post list?
// add this lineimport"./stylesheets/application.scss";
Speaking of fade effects: Let's add a second page in order to show, how you can use Matestacks app and transition component in order to implement dynamic page transitions without full browser page reload and without adding any JavaScript!
Implement dynamic page transitions
We will create a profile page in order to save the username in a session cookie rather than asking for the username on the post form! Obviously, you would use proper user management via something like devise in a real world example!
# ...# POST /postsdefcreate @post =Post.new(post_params) @post.username= cookies[:username] # add this# check if the username is already setif cookies[:username].blank?# if not complain! render json: {message:'No username given!' },status::unprocessable_entityelse# if yes, perform the code we already gotif @post.saveActionCable.server.broadcast('matestack_ui_core', {event:'cable__created_post',data:matestack_component(:post_component,post: @post) }) render json: {message:'Post was successfully created.' },status::createdelse render json: {errors: @post.errors,message:'Post could not be created.' },status::unprocessable_entityendendend# ...
classTwitterClone::Pages::Profile::Edit<Matestack::Ui::Pagedefresponse div class:"mb-3 p-3 rounded shadow-sm"do heading size:4,text:"Your Profile",class:"mb-3" form form_config_helper do div class:"mb-3"do form_input key::username,type::text,placeholder:"Username",class:"form-control",init: current_usernameend div class:"mb-3"do form_submit do button type::submit,class:"btn btn-primary",text:"Save!"endendendendendprivatedefform_config_helper {for::profile,path: profile_update_path,method::put,success: { emit:"submitted" },failure: { emit:"form_failed" },errors: { wrapper: { tag::div,class:'invalid-feedback' },input: { class:'is-invalid' } } }endend
config/routes.rb
Rails.application.routes.drawdo resources :postsdo member do put 'like',to:'posts#like'endend scope :profile,as::profiledo get 'edit',to:'profile#edit' put 'update',to:'profile#update'endend
touchapp/controllers/profile_controller.rb
app/controllers/profile_controller.rb
classProfileController<ApplicationController matestack_app TwitterClone::App# GET /profile/editdefedit render TwitterClone::Pages::Profile::Editend# PUT /profile/updatedefupdateif profile_params[:username].blank? render json: {message:'Profile could not be updated.',errors: { username: ["can't be blank!"] } },status::unprocessable_entityelse cookies[:username] = profile_params[:username] render json: {message:'Profile was successfully updated.' },status::createdendendprivate# Only allow a list of trusted parameters through.defprofile_params params.require(:profile).permit(:username)endend
app/matestack/twitter_clone/app.rb
classTwitterClone::App<Matestack::Ui::Appdefresponse div class:"container"do# heading size: 1, text: "Twitter Clone", class: "mb-5"# yield_page heading size:1,text:"Twitter Clone" transition path: posts_path do button class:"btn btn-light",text:"Timeline"end transition path: profile_edit_path do button class:"btn btn-light",text:"Your Profile"end div class:"mt-5"do yield_pageend# add the toggle components here, this way all pages are able to trigger them! toggle show_on:"submitted",hide_after:5000do div class:"container fixed-bottom w-100 bg-success text-white p-3 rounded-top"do heading size:4,text:"Success: {{ event.data.message }}"endend toggle show_on:"form_failed",hide_after:5000do div class:"container fixed-bottom w-100 bg-danger text-white p-3 rounded-top"do heading size:4,text:"Error: {{ event.data.message }}"endendendendend
Test the current state
Great, we just added a second page and added some transition components to our app and without further effort, we implemented dynamic page transitions without touching any JavaScript. The transition component triggered the app to request the desired page at the server targeting the appropriate controller action through Rails routing and adjusted the DOM where we placed the yield_page on our app!
classTwitterClone::App<Matestack::Ui::Appdefresponse div class:"container"do heading size:1,text:"Twitter Clone"# transition path: posts_path do transition path: posts_path,delay:300do button class:"btn btn-light",text:"Timeline"end# transition path: profile_edit_path do transition path: profile_edit_path,delay:300do button class:"btn btn-light",text:"Your Profile"end div class:"mt-5"do yield_pageendendendend
Test the current state
And now we do something, what's not possible in Twitter: Editing. Tweets. Inline. In pure Ruby! (Just because it's nice to showcase that)
Inline Editing
app/matestack/components/post.rb
classComponents::Post<Matestack::Ui::Component requires :postdefresponse div class:"mb-3 p-3 rounded shadow-sm",id:"post-#{post.id}"do heading size:5do plain post.username small text: post.created_at.strftime("%d.%m.%Y %H:%M")end toggle hide_on:"edit-post-#{post.id}",show_on:"updated",init_show:truedo show_partialend toggle show_on:"edit-post-#{post.id}",hide_on:"updated"do edit_partialend# paragraph text: post.body, class: "mb-5"# action path: like_post_path(post), method: :put do# button class: "btn btn-light" do# plain "Like (#{post.likes_count})"# end# endendendprivatedefshow_partial paragraph text: post.body,class:"mb-5" action path:like_post_path(post),method::putdo button class:"btn btn-light"do plain "Like (#{post.likes_count})"endend# onclick emits an event triggering the toggle components to show/hide# we use Bootstraps "d-inline" utility class here because onclick renders# a block element (will be changed to an inline element in a future release) onclick emit:"edit-post-#{post.id}",class:"d-inline"do button class:"btn btn-link"do plain "Edit"endendenddefedit_partial form form_config_helper do div class:"mb-3"do form_input key::body,type::text,placeholder:"What's up?",class:"form-control"end div class:"mb-3"do form_submit do button type::submit,class:"btn btn-primary",text:"Update!"endendendenddefform_config_helper {for: post,path:post_path(id: post.id),method::put,success: { emit:"updated" },failure: { emit:"form_failed" },errors: { wrapper: { tag::div,class:'invalid-feedback' },input: { class:'is-invalid' } } }endend
app/controller/posts_controller.rb
# ...# PUT /posts/1defupdate @post =Post.find(params[:id])if @post.update(post_params)ActionCable.server.broadcast('matestack_ui_core', {event:"cable__updated_post",data:matestack_component(:post_component,post: @post) }) render json: {message:'Post was successfully updated.' },status::createdelse render json: {errors: @post.errors,message:'Post could not be updated.' },status::unprocessable_entityendend# ...
Test the current state
Conclusion
We've built a reactive Twitter clone in pure Ruby. Fantastic! :)
Like it? Consider giving the project a star or even become a sponsor on Github, share it with your friends and colleagues (and family?) and join our Discord server! We're super happy about feedback, looking forward to hear your success stories and help you to build awesome things with Matestack!