Files
webawesome/docs/pages/components/rating.md
2023-12-05 17:37:06 -05:00

6.1 KiB

meta, layout
meta layout
title description
Rating Ratings give users a way to quickly view and provide feedback.
component
<wa-rating label="Rating"></wa-rating>
import WaRating from '@shoelace-style/shoelace/dist/react/rating';

const App = () => <WaRating label="Rating" />;

Examples

Labels

Ratings are commonly identified contextually, so labels aren't displayed. However, you should always provide one for assistive devices using the label attribute.

<wa-rating label="Rate this component"></wa-rating>
import WaRating from '@shoelace-style/shoelace/dist/react/rating';

const App = () => <WaRating label="Rate this component" />;

Maximum Value

Ratings are 0-5 by default. To change the maximum possible value, use the max attribute.

<wa-rating label="Rating" max="3"></wa-rating>
import WaRating from '@shoelace-style/shoelace/dist/react/rating';

const App = () => <WaRating label="Rating" max={3} />;

Precision

Use the precision attribute to let users select fractional ratings.

<wa-rating label="Rating" precision="0.5" value="2.5"></wa-rating>
import WaRating from '@shoelace-style/shoelace/dist/react/rating';

const App = () => <WaRating label="Rating" precision={0.5} value={2.5} />;

Symbol Sizes

Set the --symbol-size custom property to adjust the size.

<wa-rating label="Rating" style="--symbol-size: 2rem;"></wa-rating>

{% raw %}

import WaRating from '@shoelace-style/shoelace/dist/react/rating';

const App = () => <WaRating label="Rating" style={{ '--symbol-size': '2rem' }} />;

{% endraw %}

Readonly

Use the readonly attribute to display a rating that users can't change.

<wa-rating label="Rating" readonly value="3"></wa-rating>
import WaRating from '@shoelace-style/shoelace/dist/react/rating';

const App = () => <WaRating label="Rating" readonly value={3} />;

Disabled

Use the disable attribute to disable the rating.

<wa-rating label="Rating" disabled value="3"></wa-rating>
import WaRating from '@shoelace-style/shoelace/dist/react/rating';

const App = () => <WaRating label="Rating" disabled value={3} />;

Detecting Hover

Use the wa-hover event to detect when the user hovers over (or touch and drag) the rating. This lets you hook into values as the user interacts with the rating, but before they select a value.

The event has a payload with phase and value properties. The phase property tells when hovering starts, moves to a new value, and ends. The value property tells what the rating's value would be if the user were to commit to the hovered value.

<div class="detect-hover">
  <wa-rating label="Rating"></wa-rating>
  <span></span>
</div>

<script>
  const rating = document.querySelector('.detect-hover > wa-rating');
  const span = rating.nextElementSibling;
  const terms = ['No rating', 'Terrible', 'Bad', 'OK', 'Good', 'Excellent'];

  rating.addEventListener('wa-hover', event => {
    span.textContent = terms[event.detail.value];

    // Clear feedback when hovering stops
    if (event.detail.phase === 'end') {
      span.textContent = '';
    }
  });
</script>

<style>
  .detect-hover span {
    position: relative;
    top: -4px;
    left: 8px;
    border-radius: var(--wa-corners-s);
    background: var(--wa-color-neutral-spot);
    color: var(--wa-color-neutral-text-on-spot);
    text-align: center;
    padding: 4px 6px;
  }

  .detect-hover span:empty {
    display: none;
  }
</style>
import { useState } from 'react';
import WaRating from '@shoelace-style/shoelace/dist/react/rating';

const terms = ['No rating', 'Terrible', 'Bad', 'OK', 'Good', 'Excellent'];
const css = `
  .detect-hover span {
    position: relative;
    top: -4px;
    left: 8px;
    border-radius: var(--wa-corners-s);
    background: var(--wa-color-neutral-spot);
    color: var(--wa-color-neutral-text-on-spot);
    text-align: center;
    padding: 4px 6px;
  }

  .detect-hover span:empty {
    display: none;
  }
`;

function handleHover(event) {
  rating.addEventListener('wa-hover', event => {
    setFeedback(terms[event.detail.value]);

    // Clear feedback when hovering stops
    if (event.detail.phase === 'end') {
      setFeedback('');
    }
  });
}

const App = () => {
  const [feedback, setFeedback] = useState(true);

  return (
    <>
      <div class="detect-hover">
        <WaRating label="Rating" onWaHover={handleHover} />
        <span>{feedback}</span>
      </div>
      <style>{css}</style>
    </>
  );
};

Custom Icons

You can provide custom icons by passing a function to the getSymbol property.

<wa-rating label="Rating" class="rating-hearts" style="--symbol-color-active: #ff4136;"></wa-rating>

<script>
  const rating = document.querySelector('.rating-hearts');
  rating.getSymbol = () => '<wa-icon name="heart" variant="solid"></wa-icon>';
</script>

{% raw %}

import WaRating from '@shoelace-style/shoelace/dist/react/rating';

const App = () => (
  <WaRating
    label="Rating"
    getSymbol={() => '<wa-icon name="heart" variant="solid"></wa-icon>'}
    style={{ '--symbol-color-active': '#ff4136' }}
  />
);

{% endraw %}

Value-based Icons

You can also use the getSymbol property to render different icons based on value.

<wa-rating label="Rating" class="rating-emojis"></wa-rating>

<script>
  const rating = document.querySelector('.rating-emojis');

  rating.getSymbol = value => {
    const icons = ['face-angry', 'face-frown', 'face-meh', 'face-smile', 'face-laugh'];
    return `<wa-icon name="${icons[value - 1]}"></wa-icon>`;
  };
</script>
import WaRating from '@shoelace-style/shoelace/dist/react/rating';

function getSymbol(value) {
  const icons = ['face-angry', 'face-frown', 'face-meh', 'face-smile', 'face-laugh'];
  return `<wa-icon name="${icons[value - 1]}"></wa-icon>`;
}

const App = () => <WaRating label="Rating" getSymbol={getSymbol} />;