Movatterモバイル変換


[0]ホーム

URL:


Skip to content
DEV Community
Log in Create account

DEV Community

Cover image for Hanami: Unobtrusive JavaScript
Jan Bajena
Jan Bajena

Posted on • Edited on

     

Hanami: Unobtrusive JavaScript

Intro

Inone of my previous posts I wrote down the impressions about Hanami I've had while working on my side project,Flashcard Genius. One of the things I mentioned there was that some features you'd expect from a framework are missing or underdocumented. One of such features I missed in Hanami was unobtrusive javascript.

Rails comes out-of-the-box withhelpers that allow developers to do following things without writing a single JavaScript line:

  • Create links to POST/DELETE/PATCH endpoints (link_to "Delete article, @article, method: :delete)
  • Send forms remotely using AJAX just by addingremote: true to your form tag.
  • Show simple confirmation dialog after clicking a link (link_to "Dangerous zone", dangerous_zone_path, data: { confirm: 'Are you sure?' })
  • Disable submit buttons while waiting for the form to be sent (f.submit data: { "disable-with": "Saving..." })

In my application I had a need for remote DELETE links (removing flashcards, log out endpoint) and remote forms (quickly add/edit a card without reloading the page), so I started looking for solutions and... turns out there's a way to do it in Hanami thanks to a plugin calledhanami-ujs. Surprisingly, this library is hosted in the official Hanami GitHub organization, but the documentation almost doesn't exist. In this post I'm gonna show you how I solved some of my problems using this library.

Installation

Installation and setup is well described in gem'sREADME file, have a look there.

Use case #1 - DELETE links

When you're following RESTful resource routing then there's a 99% chance that a route to destroy a resource will be aDELETE method route. Unfortunately Hanami doesn't support creating links to DELETE paths out-of-the box, so e.g. my logout link looked like this under the hood:

<%=  form_for :session, routes.logout_path, method: :delete do    a href: "#", onclick: "this.closest('form').submit();return false;" do      "Logout"    end  end%>
Enter fullscreen modeExit fullscreen mode

Even though it worked it was a nasty hack. It was also violatingunsafe-inline directive from Content Security Policy.

After installinghanami-ujs this is one line is all I need instead:

<%= link_to "Logout", routes.logout_path, "data-method": :delete %>
Enter fullscreen modeExit fullscreen mode

Use case #2 - Remote forms

Sometimes you want your view to be a bit more dynamic and don't want to reload the whole page when a form is sent. In this case AJAX is your friend.hanami-ujs offers sending remote AJAX forms. I used this feature e.g. on the word edit form in Flashcard Genius. You can see how that works on the GIF below:

screencast 2020-11-08 12-37-52

In order to achieve this result I needed two things - form definition and JS event handlers.

Form definition

If you know a bit of Hanami you'll notice that this is a standard Hanami form with two additions:

  • remote: true - tellshanami-ujs that the form will be sent by AJAX
  • "data-method": :patch option - method thathanami-ujs will use to send the form
<%=  form_for(:word, routes.word_path(word.id), remote: true, method: :patch, "data-method": :patch, class: "edit-word-form") do    div class: "card-body" do      div class: "row margin-bottom-none" do        div class: "col xs-12" do          div class: "form-group" do            label      :question            text_field :question, class: "input-block", required: true, value: word.question          end        end        div class: "col xs-12" do          div class: "form-group" do            label      :question_example            text_field :question_example, class: "input-block", value: word.question_example          end        end        div class: "col xs-12" do          div class: "form-group" do            label      :answer            text_field :answer, class: "input-block", required: true, value: word.answer          end        end        div class: "col xs-12" do          div class: "form-group" do            label      :answer_example            text_field :answer_example, class: "input-block", value: word.answer_example          end        end      end    end    div class: "card-footer" do      div class: "row margin-none padding-none flex-edges" do        span("Cancel", class: "paper-btn btn-small btn-primary-outline cancel-edit-button margin-none")        submit("Save", class: "btn-secondary-outline btn-small edit-save-button margin-none", "data-disable-with": "Saving...")      end    end  end%>
Enter fullscreen modeExit fullscreen mode

JS event handler

In order to be able to dynamically react on AJAX response we need some JavaScript on our page.hanami-ujs defines two event types:

  • ajax:before - Called before the request is sent. It can be used e.g. to clear forms errors after previous request.
  • ajax:complete - Called when the response is received (no matter if the request succeeded or not).

Here's how my JS code for the edit word form looks like. The server updates the word and returns a HTML template with the updated record. The new HTML replaces the old one:

functionupdateCompleteHandler(event){varform=event.target;// There can be other `ajax:complete` handlers defined on the page.// Make sure that this code is executed when `edit-word-form` is sent.if(!form.className.includes("edit-word-form")){return;}varstatus=event.detail.status;if(status>=200&&status<300){alertify.success("Word updated");varcard=form.closest(".flashcard-column");// Replace the current word with server's responsecard.outerHTML=event.detail.response;}else{alertify.error("Update failed");varsaveButton=form.getElementsByClassName("edit-save-button")[0];saveButton.disabled=false;saveButton.innerText="Save"}}document.addEventListener("ajax:complete",updateCompleteHandler);
Enter fullscreen modeExit fullscreen mode

Additional useful features

hanami-ujs has also two other useful features:

  • You can add"disable-with": "Saving..." to your form submit buttons. This'll automatically disable the button to prevent double-sends and replace the its text withSaving.... One drawback of this feature is that it wasn't created with AJAX forms in mind, so when you want to itdisable-with with remote forms you'll have to re-enable the button and change its text back to original after the request. You can find the example in previous snippets.
  • You can show a simple confirmation dialog that prevents sending a form/following a link by adding"data-confirm": "Are you sure you want to do it?". This is how one of the buttons in my app looks like:
<%=    link_to(      "",      routes.word_list_path(word_list.id),      id: "delete-list-button",      class: "paper-btn btn-danger-outline btn-small margin-none fas fa-trash-alt",      "data-method": :delete,      "data-confirm": "Are you sure you want to delete this list?",      "data-disable-with": "",      title: "Delete list"    )  %>
Enter fullscreen modeExit fullscreen mode

which results in a following message:

image

Summary

Even thoughhanami-ujs may seem quite abandoned it provides tools that save a lot of time when creating simple server-rendered apps. If you're creating such an app in Hanami there's a big chance you'll want to use it. For morehanami-ujs examples (and Hanami in general) you can check Flashcard Genius' repository athttps://github.com/Bajena/flashcard-genius.

Happy coding :)

Top comments(0)

Subscribe
pic
Create template

Templates let you quickly answer FAQs or store snippets for re-use.

Dismiss

Are you sure you want to hide this comment? It will become hidden in your post, but will still be visible via the comment'spermalink.

For further actions, you may consider blocking this person and/orreporting abuse

Not-a-rockstar, but passionate Ruby developer based in Warsaw, Poland.
  • Location
    Warsaw, Poland
  • Education
    Warsaw University of Technology
  • Work
    Ruby developer at Productboard
  • Joined

More fromJan Bajena

DEV Community

We're a place where coders share, stay up-to-date and grow their careers.

Log in Create account

[8]ページ先頭

©2009-2025 Movatter.jp