ruby-on-railsformsbootstrap-5ratingrails7

How to add star rating input field review form, Rails 7, Bootstrap (no jquery)?


I am building a Rails 7 app using bootstrap (without jquery), in which users can leave a review for various parks. One of the review fields is a 5-star rating.

Currently I have the form input as a text_field, but I would like it to be 5 stars that the user can click on to select the rating, then submit together with the review form.

I am already displaying the park ratings after they've been submitted, but I am stuck on the input display.

I thought maybe I can somehow change the UI for a range_field or collection_radio_buttons, but I can't figure out how to make it work...

Park model

class Park < ApplicationRecord
  has_many :visits, dependent: :destroy
  has_many :visited_users, through: :visits, source: :user

  has_many :favorites, dependent: :destroy
  has_many :favorited_users, through: :favorites, source: :user

  has_many :reviews, as: :reviewable

  include Translatable
  translates :name, :website_url

  validates :name_en, presence: true
  validates :name_he, presence: true
  validates :region, presence: true
  validates :latitude, presence: true
  validates :longitude, presence: true

  enum :region, { north: 0, center: 1, south: 2, west_bank: 3 }
  enum :park_system, { kkl_jnf: 0, inpa: 1 },
                      default: :inpa

  def default_image
    Review&.where(reviewable: self)&.left_joins(:images_attachments)&.where&.not(active_storage_attachments: { id: nil })&.first&.images&.first
  end

  def favorited_by?(user)
    return if user.nil?

    favorited_users.include?(user)
  end

  def visited_by?(user)
    return if user.nil?

    visited_users.include?(user)
  end

end

Review model

class Review < ApplicationRecord
  has_many_attached :images, dependent: :destroy

  # validates :title, presence: true
  validates :body, presence: true
  validates :rating, presence: true, numericality:
            { greater_than_or_equal_to: 1, less_than_or_equal_to: 5,
              only_integer: true }

  belongs_to :reviewable, polymorphic: true, counter_cache: true
  belongs_to :user

  after_commit :update_reviewable_rating, on: [:create, :update, :destroy]

  def update_reviewable_rating
    reviewable.update! average_rating: reviewable.reviews.average(:rating)
    # avg + (rating - avg) / count
  end

end

Reviews controller

# frozen_string_literal: true

class ReviewsController < ApplicationController
  before_action :authenticate_user!
  before_action :find_park
  before_action :find_review, only: %i[ edit update destroy ]

  def new
    @review = Review.new(reviewable: @park)
  end

  def create
    @review = @park.reviews.new(review_params)
    @review.reviewable_id = @park.id
    @review.user_id = current_user.id

    respond_to do |format|
      if @review.save
        format.html { redirect_to params[:previous_request], notice: "Your review for #{@park.name} was successfully added." }
        format.json { render :show, status: :created, location: @park }
      else
        format.html { render :new, status: :unprocessable_entity }
        format.json { render json: @review.errors, status: :unprocessable_entity }
      end
    end
  end

  def edit
  end

  def update
    respond_to do |format|
      if @review.update(review_params)
        format.html { redirect_to params[:previous_request], notice: "Your review for #{@park.name} was successfully updated." }
        format.json { render :show, status: :ok, location: @park }
      else
        format.html { render :edit, status: :unprocessable_entity }
        format.json { render json: @review.errors, status: :unprocessable_entity }
      end
    end
  end

  def destroy
    @review.destroy

    respond_to do |format|
      format.html { redirect_to params[:previous_request], notice: "Your review for #{@park.name} was successfully deleted." }
      format.json { head :no_content }
    end
  end

  private

    def review_params
      params.require(:review).permit(:rating, :body, images: [])
    end

    def find_park
      @park = Park.find(params[:park_id])
    end

    def find_review
      @review = Review.find(params[:id])
    end

end

Parks show view, reviews partial

<% @park.reviews.includes(:user).order("updated_at desc").each do |review| %>
  <div class="col">
    <div class="card px-0 mt-4 mb-4 border-0">
      <div class="card-body">
        <h5 class="card-title mb-4">
          <%= gravatar_for review.user, size: 50 %>
          <div class="strong">
            <%= link_to review.user.name, review.user, class: "link-dark" %>
          </div>
        </h5>
        <h6 class="card-subtitle mt-2">
          <%
            review_star_classes = ["#DEDEDE", "#DEDEDE", "#DEDEDE", "#DEDEDE", "#DEDEDE"]

            review.rating.times do |i|
              review_star_classes[i] = "#fbbc04"
            end
          %>

          <% review_star_classes.each do |star_class| %>
            <svg
              xmlns="http://www.w3.org/2000/svg"
              viewBox="0 0 24 24"
              fill="<%= star_class %>"
              width="16"
              height="16"
              class="w-5 h-5">
              <path
                fill-rule="evenodd"
                d="M10.788 3.21c.448-1.077 1.976-1.077 2.424 0l2.082 5.007 5.404.433c1.164.093 1.636 1.545.749 2.305l-4.117 3.527 1.257 5.273c.271 1.136-.964 2.033-1.96 1.425L12 18.354 7.373 21.18c-.996.608-2.231-.29-1.96-1.425l1.257-5.273-4.117-3.527c-.887-.76-.415-2.212.749-2.305l5.404-.433 2.082-5.006z"
                clip-rule="evenodd" />
            </svg>
          <% end %>
        </h6>
        <div class="card-text mt-4">
          <div>
            <%= review.updated_at.to_fs(:long) %>
          </div>
          <div class="mt-2">
            <%= simple_format(review.body) %>
          </div>
          <% if current_user && current_user === review.user %>
            <div>
              <%= link_to "Edit", edit_park_review_path(review.reviewable, review) %>
              <%= button_to "Delete", park_review_path(review.reviewable, review),
                                      class: "btn btn-link p-0 m-0 d-inline align-baseline text-decoration-none",
                                      form: {data: { turbo_confirm: "Are you sure?"} },
                                      method: :delete %>
            </div>
          <% end %>
        </div>
      </div>
    </div>
  </div>
<% end %>

New review form

<%= form_with model: @review, url: park_reviews_path, local:true do |f| %>
  <%= hidden_field_tag :previous_request, request.referer %>

  <div class="field form-group">
    <%= f.label :rating, "Rate your experience", style: "display: block" %>
    <%= f.text_field :rating, class: "form-control",
                          placeholder: "From 1 to 5 stars",
                          autofocus: true %>
  </div>
  <div class="field form-group">
    <%= f.text_area :body, class: "form-control",
                        placeholder: "Tell people about your experience" %>
  </div>
  <div class="field form-group">
    <%= f.file_field :images, multiple: true %>
  </div>
  <div class="actions">
    <%= f.submit "Submit your review", class: "btn btn-primary" %>
  </div>
<% end %>


Solution

  • I'm not sure this was the best solution but I ended up with the following code, if it helps someone in the future. Definitely not perfect (for example, I had to make the label text white, since I couldn't figure out a way to display an empty label).

    New review form

    <%= form_with model: @review, url: park_reviews_path, local:true do |f| %>
      <%= hidden_field_tag :previous_request, request.referer %>
    
      <div class="field form-group">
      <%= f.label :rating, "What's your overall rating?" %>
        <div class="rating">
          <%= f.collection_radio_buttons(:rating, [[5],[4],[3],[2],[1]], :first, :last) do |star| %>
            <%= star.radio_button %>
            <%= star.label(class: "text-white") %>
          <% end %>
        </div>
      </div>
      <div class="field form-group">
        <%= f.label :body, "Leave a review" %>
        <%= f.text_area :body, class: "form-control",
                            placeholder: "Tell people about your experience" %>
      </div>
      <div class="field form-group">
        <%= f.label :images, "Add some photos" %>
        <%= f.file_field :images, multiple: true %>
      </div>
      <div class="actions">
        <%= f.submit "Submit your review", class: "btn btn-primary" %>
      </div>
    <% end %>
    

    CSS

    
    /*
    * star rating
    */
    
    .rating {
      display: flex;
      width: 100%;
      justify-content: left;
      overflow: hidden;
      flex-direction: row-reverse;
      // height: 150px;
      position: relative;
    }
    
    .rating-0 {
      filter: grayscale(100%);
    }
    
    .rating > input {
      display: none;
    }
    
    .rating > label {
      cursor: pointer;
      width: 40px;
      height: 40px;
      margin-top: auto;
      background-image: url("data:image/svg+xml;charset=UTF-8,%3csvg xmlns='http://www.w3.org/2000/svg' width='126.729' height='126.73'%3e%3cpath fill='%23dedede' d='M121.215 44.212l-34.899-3.3c-2.2-.2-4.101-1.6-5-3.7l-12.5-30.3c-2-5-9.101-5-11.101 0l-12.4 30.3c-.8 2.1-2.8 3.5-5 3.7l-34.9 3.3c-5.2.5-7.3 7-3.4 10.5l26.3 23.1c1.7 1.5 2.4 3.7 1.9 5.9l-7.9 32.399c-1.2 5.101 4.3 9.3 8.9 6.601l29.1-17.101c1.9-1.1 4.2-1.1 6.1 0l29.101 17.101c4.6 2.699 10.1-1.4 8.899-6.601l-7.8-32.399c-.5-2.2.2-4.4 1.9-5.9l26.3-23.1c3.8-3.5 1.6-10-3.6-10.5z'/%3e%3c/svg%3e");
      background-repeat: no-repeat;
      background-position: center;
      background-size: 76%;
      transition: .3s;
    }
    
    .rating > input:checked ~ label,
    .rating > input:checked ~ label ~ label {
      background-image: url("data:image/svg+xml;charset=UTF-8,%3csvg xmlns='http://www.w3.org/2000/svg' width='126.729' height='126.73'%3e%3cpath fill='%23fbbc04' d='M121.215 44.212l-34.899-3.3c-2.2-.2-4.101-1.6-5-3.7l-12.5-30.3c-2-5-9.101-5-11.101 0l-12.4 30.3c-.8 2.1-2.8 3.5-5 3.7l-34.9 3.3c-5.2.5-7.3 7-3.4 10.5l26.3 23.1c1.7 1.5 2.4 3.7 1.9 5.9l-7.9 32.399c-1.2 5.101 4.3 9.3 8.9 6.601l29.1-17.101c1.9-1.1 4.2-1.1 6.1 0l29.101 17.101c4.6 2.699 10.1-1.4 8.899-6.601l-7.8-32.399c-.5-2.2.2-4.4 1.9-5.9l26.3-23.1c3.8-3.5 1.6-10-3.6-10.5z'/%3e%3c/svg%3e");
    }
    
    
    .rating > input:not(:checked) ~ label:hover,
    .rating > input:not(:checked) ~ label:hover ~ label {
      background-image: url("data:image/svg+xml;charset=UTF-8,%3csvg xmlns='http://www.w3.org/2000/svg' width='126.729' height='126.73'%3e%3cpath fill='%23e0a803' d='M121.215 44.212l-34.899-3.3c-2.2-.2-4.101-1.6-5-3.7l-12.5-30.3c-2-5-9.101-5-11.101 0l-12.4 30.3c-.8 2.1-2.8 3.5-5 3.7l-34.9 3.3c-5.2.5-7.3 7-3.4 10.5l26.3 23.1c1.7 1.5 2.4 3.7 1.9 5.9l-7.9 32.399c-1.2 5.101 4.3 9.3 8.9 6.601l29.1-17.101c1.9-1.1 4.2-1.1 6.1 0l29.101 17.101c4.6 2.699 10.1-1.4 8.899-6.601l-7.8-32.399c-.5-2.2.2-4.4 1.9-5.9l26.3-23.1c3.8-3.5 1.6-10-3.6-10.5z'/%3e%3c/svg%3e");
    }