@mantine/core
Version:
React components library focused on usability, accessibility and developer experience
168 lines (167 loc) • 6.14 kB
JavaScript
"use client";
import { getSize } from "../../core/utils/get-size/get-size.mjs";
import { createVarsResolver } from "../../core/styles-api/create-vars-resolver/create-vars-resolver.mjs";
import { getThemeColor } from "../../core/MantineProvider/color-functions/get-theme-color/get-theme-color.mjs";
import { useProps } from "../../core/MantineProvider/use-props/use-props.mjs";
import { useStyles } from "../../core/styles-api/use-styles/use-styles.mjs";
import { factory } from "../../core/factory/factory.mjs";
import { Box } from "../../core/Box/Box.mjs";
import { useDirection } from "../../core/DirectionProvider/DirectionProvider.mjs";
import { RatingProvider } from "./Rating.context.mjs";
import { RatingItem } from "./RatingItem/RatingItem.mjs";
import Rating_module_default from "./Rating.module.mjs";
import { useRef, useState } from "react";
import { clamp, useId as useId$1, useMergedRef, useUncontrolled } from "@mantine/hooks";
import { jsx } from "react/jsx-runtime";
//#region packages/@mantine/core/src/components/Rating/Rating.tsx
function roundValueTo(value, to) {
const rounded = Math.round(value / to) * to;
const precision = `${to}`.split(".")[1]?.length || 0;
return Number(rounded.toFixed(precision));
}
const defaultProps = {
size: "sm",
getSymbolLabel: (value) => `${value}`,
count: 5,
fractions: 1,
color: "yellow"
};
const varsResolver = createVarsResolver((theme, { size, color }) => ({ root: {
"--rating-size": getSize(size, "rating-size"),
"--rating-color": getThemeColor(color, theme)
} }));
const Rating = factory((_props) => {
const props = useProps("Rating", defaultProps, _props);
const { classNames, className, style, styles, unstyled, vars, name, id, value, defaultValue, onChange, fractions, count, onMouseEnter, readOnly, allowClear, onMouseMove, onHover, onMouseLeave, onTouchStart, onTouchEnd, size, variant, getSymbolLabel, color, emptySymbol, fullSymbol, highlightSelectedOnly, attributes, ref, ...others } = props;
const getStyles = useStyles({
name: "Rating",
classes: Rating_module_default,
props,
className,
style,
classNames,
styles,
unstyled,
attributes,
vars,
varsResolver
});
const { dir } = useDirection();
const _name = useId$1(name);
const _id = useId$1(id);
const rootRef = useRef(null);
const [_value, setValue] = useUncontrolled({
value,
defaultValue,
finalValue: 0,
onChange
});
const [hovered, setHovered] = useState(-1);
const [isOutside, setOutside] = useState(true);
const _fractions = Math.floor(fractions);
const _count = Math.floor(count);
const decimalUnit = 1 / _fractions;
const stableValueRounded = roundValueTo(_value, decimalUnit);
const finalValue = hovered !== -1 ? hovered : stableValueRounded;
const getRatingFromCoordinates = (x) => {
if (!rootRef.current) return 0;
const { left, right, width } = rootRef.current.getBoundingClientRect();
const symbolWidth = width / _count;
return clamp(roundValueTo((dir === "rtl" ? right - x : x - left) / symbolWidth + decimalUnit / 2, decimalUnit), decimalUnit, _count);
};
const handleMouseEnter = (event) => {
onMouseEnter?.(event);
!readOnly && setOutside(false);
};
const handleMouseMove = (event) => {
onMouseMove?.(event);
if (readOnly) return;
const rounded = getRatingFromCoordinates(event.clientX);
setHovered(rounded);
rounded !== hovered && onHover?.(rounded);
};
const handleMouseLeave = (event) => {
onMouseLeave?.(event);
if (readOnly) return;
setHovered(-1);
setOutside(true);
hovered !== -1 && onHover?.(-1);
};
const handleTouchStart = (event) => {
const { touches } = event;
if (touches.length !== 1) return;
if (!readOnly) {
const touch = touches[0];
setValue(getRatingFromCoordinates(touch.clientX));
}
onTouchStart?.(event);
};
const handleTouchEnd = (event) => {
event.preventDefault();
onTouchEnd?.(event);
};
const handleItemBlur = () => isOutside && setHovered(-1);
const handleInputChange = (event) => {
if (!readOnly) if (typeof event === "number") setHovered(event);
else setHovered(parseFloat(event.target.value));
};
const handleChange = (event) => {
if (!readOnly) {
const newValue = typeof event === "number" ? event : parseFloat(event.target.value);
if (allowClear && newValue === stableValueRounded) setValue(0);
else setValue(newValue);
}
};
const items = Array(_count).fill(0).map((_, index) => {
const integerValue = index + 1;
const fractionItems = Array.from(new Array(index === 0 ? _fractions + 1 : _fractions));
const isGroupActive = !readOnly && Math.ceil(hovered) === integerValue;
return /* @__PURE__ */ jsx("div", {
"data-active": isGroupActive || void 0,
...getStyles("symbolGroup"),
children: fractionItems.map((__, fractionIndex) => {
const fractionValue = decimalUnit * (index === 0 ? fractionIndex : fractionIndex + 1);
const symbolValue = roundValueTo(integerValue - 1 + fractionValue, decimalUnit);
return /* @__PURE__ */ jsx(RatingItem, {
getSymbolLabel,
emptyIcon: emptySymbol,
fullIcon: fullSymbol,
full: highlightSelectedOnly ? symbolValue === finalValue : symbolValue <= finalValue,
active: symbolValue === finalValue,
checked: symbolValue === stableValueRounded,
readOnly,
fractionValue,
value: symbolValue,
name: _name,
onChange: handleChange,
onBlur: handleItemBlur,
onInputChange: handleInputChange,
id: `${_id}-${index}-${fractionIndex}`
}, `${integerValue}-${symbolValue}`);
})
}, integerValue);
});
return /* @__PURE__ */ jsx(RatingProvider, {
value: { getStyles },
children: /* @__PURE__ */ jsx(Box, {
ref: useMergedRef(rootRef, ref),
...getStyles("root"),
onMouseMove: handleMouseMove,
onMouseEnter: handleMouseEnter,
onMouseLeave: handleMouseLeave,
onTouchStart: handleTouchStart,
onTouchEnd: handleTouchEnd,
variant,
size,
id: _id,
...others,
children: items
})
});
});
Rating.classes = Rating_module_default;
Rating.varsResolver = varsResolver;
Rating.displayName = "@mantine/core/Rating";
//#endregion
export { Rating };
//# sourceMappingURL=Rating.mjs.map