@dnb/eufemia
Version:
DNB Eufemia Design System UI Library
402 lines (401 loc) • 13 kB
JavaScript
"use client";
import _extends from "@babel/runtime-corejs3/helpers/esm/extends";
import React, { useCallback } from 'react';
import classnames from 'classnames';
import { cleanNumber, getCurrencySymbol } from "../number-format/NumberUtils.js";
import { isTrue, dispatchCustomElementEvent, extendPropsWithContext, keycode } from "../../shared/component-helper.js";
import { safeSetSelection } from "./text-mask/createTextMaskInputElement.js";
import { isNumber } from "./text-mask/utilities.js";
import TextMask from "./TextMask.js";
import createNumberMask from "./addons/createNumberMask.js";
import InputMaskedContext from "./InputMaskedContext.js";
import { isRequestingLocaleSupport, isRequestingNumberMask, correctNumberValue, handlePercentMask, handleCurrencyMask, handleNumberMask, correctCaretPosition, getSoftKeyboardAttributes, handleThousandsSeparator, handleDecimalSeparator, fromJSON, invisibleSpace } from "./InputMaskedUtils.js";
import { useIsomorphicLayoutEffect as useLayoutEffect } from "../../shared/helpers/useIsomorphicLayoutEffect.js";
const NUMBER_MINUS = '-|−|‐|‒|–|—|―';
export const useFilteredProps = () => {
const {
props
} = React.useContext(InputMaskedContext);
const {
mask,
number_mask,
currency_mask,
number_format,
mask_options,
as_currency,
as_number,
as_percent,
locale,
show_mask,
show_guide,
pipe,
keep_char_positions,
placeholder_char,
...attributes
} = props;
return {
props,
htmlAttributes: Object.freeze(attributes)
};
};
export const useTranslation = () => {
const {
props,
context
} = React.useContext(InputMaskedContext);
let {
locale
} = props;
if (!locale && context !== null && context !== void 0 && context.locale) {
locale = context.locale;
}
return locale;
};
export const useLocalValue = () => {
const {
props,
context
} = React.useContext(InputMaskedContext);
const maskParams = useNumberMaskParams() || {};
const locale = useTranslation();
const [localValue, setLocalValue] = React.useState(() => {
return correctNumberValue({
locale,
props,
maskParams
});
});
React.useEffect(() => {
const value = correctNumberValue({
localValue,
locale,
props,
maskParams
});
setLocalValue(value);
}, [props, context, locale]);
return {
localValue,
setLocalValue
};
};
export const useNumberMask = () => {
const maskParams = useNumberMaskParams();
const {
props
} = React.useContext(InputMaskedContext);
if (!maskParams || !isRequestingNumberMask(props)) {
return null;
}
const mask = createNumberMask(maskParams);
mask.maskParams = maskParams;
return mask;
};
export const useMask = () => {
const {
props
} = React.useContext(InputMaskedContext);
const numberMask = useNumberMask();
if (numberMask) {
return numberMask;
}
return props.mask;
};
export const useMaskParams = () => {
const {
props
} = React.useContext(InputMaskedContext);
const {
keep_char_positions,
show_guide,
show_mask,
placeholder_char,
placeholder
} = props;
const mask = useMask();
const maskParams = useNumberMaskParams() || {};
maskParams.showMask = !placeholder && isTrue(show_mask);
maskParams.placeholderChar = placeholder_char;
if (typeof (mask === null || mask === void 0 ? void 0 : mask.placeholderChar) !== 'undefined') {
maskParams.placeholderChar = mask.placeholderChar;
}
if (maskParams.placeholderChar === null) {
maskParams.placeholderChar = invisibleSpace;
}
if (typeof (mask === null || mask === void 0 ? void 0 : mask.showMask) !== 'undefined') {
maskParams.showMask = mask.showMask;
}
maskParams.showGuide = isTrue(show_guide);
maskParams.keepCharPositions = isTrue(keep_char_positions);
return maskParams;
};
export const useInputElement = () => {
const {
props
} = React.useContext(InputMaskedContext);
const {
pipe,
inner_ref
} = props;
const mask = useMask();
const {
showMask,
showGuide,
placeholderChar,
keepCharPositions
} = useMaskParams();
const isFn = typeof inner_ref === 'function';
const refHook = React.useRef(null);
const ref = !isFn && inner_ref || refHook;
useLayoutEffect(() => {
if (isFn) {
inner_ref === null || inner_ref === void 0 || inner_ref(ref.current);
}
}, [inner_ref, isFn, ref]);
const inputElementRef = React.useRef(React.createElement("input", {
ref: ref
}));
return useCallback((params, innerRef) => {
innerRef.current = ref.current;
return React.createElement(TextMask, _extends({
inputRef: ref,
inputElement: inputElementRef.current,
pipe: pipe,
mask: mask || createNumberMask(),
showMask: showMask,
guide: showGuide,
keepCharPositions: keepCharPositions,
placeholderChar: placeholderChar
}, getSoftKeyboardAttributes(mask) || {}, params, {
className: classnames(params.className, showMask && showGuide && placeholderChar && placeholderChar !== invisibleSpace && 'dnb-input-masked--guide')
}));
}, [keepCharPositions, mask, pipe, placeholderChar, ref, showGuide, showMask]);
};
export const useEventMapping = ({
setLocalValue
}) => {
const callEvent = useCallEvent({
setLocalValue
});
return {
onBeforeInput: event => callEvent({
event
}, 'onBeforeInput'),
onInput: event => callEvent({
event
}, 'onInput'),
onFocus: params => callEvent(params, 'on_focus'),
onBlur: params => callEvent(params, 'on_blur'),
onMouseUp: event => callEvent({
event
}, 'on_mouse_up'),
onMouseDown: event => callEvent({
event
}, 'on_mouse_down'),
onKeyDown: params => callEvent(params, 'on_key_down'),
onSubmit: params => callEvent(params, 'on_submit'),
onChange: params => callEvent(params, 'on_change'),
on_focus: undefined,
on_blur: undefined,
on_key_down: undefined,
on_submit: undefined,
on_change: undefined
};
};
const useCallEvent = ({
setLocalValue
}) => {
const maskParamsRef = React.useRef(null);
maskParamsRef.current = useMaskParams();
const {
props
} = React.useContext(InputMaskedContext);
const isNumberMask = useNumberMask();
const decimalSeparators = /[,.'·]/;
let isUnidentified = false;
const callEvent = ({
event,
value
}, name) => {
const maskParams = maskParamsRef.current;
value = value || event.target.value;
const selStart = event.target.selectionStart;
let keyCode = keycode(event);
if (name === 'on_key_down' && (event.which === 229 || keyCode === undefined)) {
isUnidentified = true;
}
if (isUnidentified && name === 'onBeforeInput' && typeof (event === null || event === void 0 ? void 0 : event.data) !== 'undefined') {
name = 'on_key_down';
keyCode = event.data;
isUnidentified = false;
}
if (isUnidentified && event.key === '0') {
keyCode = '0';
isUnidentified = false;
}
if (maskParams !== null && maskParams !== void 0 && maskParams.disallowLeadingZeroes && (name === 'onInput' || name === 'on_blur')) {
const isNegative = new RegExp(`^${NUMBER_MINUS}`, 'g').test(value);
if ((isNegative ? selStart > 1 : selStart > 0) || name === 'on_blur') {
const onlyNumber = value.replace(new RegExp(`[^\\d${maskParams.decimalSymbol}]`, 'g'), '');
let leadingZeroes = 0;
for (let i = 0; i < onlyNumber.length - 1; i++) {
if (onlyNumber.charAt(i) === '0' && onlyNumber.charAt(i + 1) !== maskParams.decimalSymbol) {
leadingZeroes++;
} else {
break;
}
}
let newSelStart = selStart;
let newValue = value;
let firstNumberIndex = 0;
if (leadingZeroes > 0) {
for (let i = 0; i < value.length; i++) {
firstNumberIndex = i;
const char = value.charAt(i);
if (char !== '0' && isNumber(parseInt(char)) || value.charAt(i + 1) === maskParams.decimalSymbol) {
break;
}
}
newValue = value.substring(0, isNegative ? 1 : 0) + value.substring(firstNumberIndex);
newSelStart = selStart > firstNumberIndex ? selStart - (value.length - newValue.length) : isNegative ? 1 : 0;
}
if (newValue !== value) {
setLocalValue(newValue);
event.target.value = newValue;
safeSetSelection(event.target, newSelStart);
value = newValue;
}
}
}
if (name === 'on_key_down' && isNumberMask && !isUnidentified && maskParams !== null && maskParams !== void 0 && maskParams.decimalSymbol) {
const hasDecimalSymbol = value.includes(maskParams.decimalSymbol);
const allowedDecimals = maskParams.decimalLimit > 0 || maskParams.allowDecimal !== false;
if (!allowedDecimals && decimalSeparators.test(keyCode)) {
event.preventDefault();
}
const charAtSelection = value.slice(selStart, selStart + 1);
if (allowedDecimals) {
if (hasDecimalSymbol && decimalSeparators.test(keyCode)) {
if (decimalSeparators.test(charAtSelection)) {
const index = value.indexOf(maskParams.decimalSymbol);
if (index > -1) {
safeSetSelection(event.target, index + 1);
}
}
event.preventDefault();
} else if (!hasDecimalSymbol && keyCode !== maskParams.decimalSymbol && decimalSeparators.test(keyCode)) {
value = value.slice(0, selStart);
setLocalValue(value + maskParams.decimalSymbol);
event.target.value = value + maskParams.decimalSymbol;
event.preventDefault();
}
}
if (keyCode === 'delete' && charAtSelection === (maskParams.thousandsSeparatorSymbol || ' ')) {
safeSetSelection(event.target, selStart + 1);
event.preventDefault();
}
}
let num = cleanNumber(value, {
prefix: maskParams.prefix,
suffix: maskParams.suffix,
decimalSeparator: maskParams.decimalSymbol || ',',
thousandsSeparator: maskParams.thousandsSeparatorSymbol || ' '
});
if (num === '-') {
num = -0;
}
const numberValue = Number(num);
const cleanedValue = numberValue === 0 && String(num).charAt(0) !== '0' ? '' : num;
switch (name) {
case 'on_focus':
case 'on_key_down':
case 'on_mouse_down':
case 'on_mouse_up':
event.target.runCorrectCaretPosition = () => correctCaretPosition(event.target, maskParamsRef, props);
if (!event.target.__getCorrectCaretPosition) {
event.target.runCorrectCaretPosition();
}
break;
}
const result = dispatchCustomElementEvent(props, name, {
event,
value,
numberValue,
cleanedValue
});
if (name === 'on_change') {
setLocalValue(value);
}
return result;
};
return callEvent;
};
const useNumberMaskParams = () => {
var _currency_mask;
const {
props
} = React.useContext(InputMaskedContext);
const locale = useTranslation();
if (!isRequestingNumberMask(props)) {
return {
...fromJSON(props.mask_options)
};
}
let {
number_mask,
currency_mask,
mask_options
} = props;
const {
as_number,
as_percent,
as_currency,
value
} = props;
mask_options = fromJSON(mask_options);
number_mask = isTrue(number_mask) ? {} : fromJSON(number_mask);
currency_mask = isTrue(currency_mask) ? {} : fromJSON(currency_mask, {
currency: currency_mask
});
if (!((_currency_mask = currency_mask) !== null && _currency_mask !== void 0 && _currency_mask.currency)) {
delete currency_mask.currency;
}
if (isRequestingLocaleSupport(props)) {
const thousandsSeparatorSymbol = handleThousandsSeparator(locale);
const decimalSymbol = handleDecimalSeparator(locale);
if (isTrue(as_number) || isTrue(as_percent)) {
number_mask = extendPropsWithContext(number_mask, null, {
decimalSymbol,
thousandsSeparatorSymbol
});
} else if (as_currency) {
var _currency_mask2;
currency_mask = extendPropsWithContext(currency_mask, null, {
decimalSymbol,
thousandsSeparatorSymbol,
currency: getCurrencySymbol(locale, typeof as_currency === 'string' ? as_currency : null, (_currency_mask2 = currency_mask) === null || _currency_mask2 === void 0 ? void 0 : _currency_mask2.currencyDisplay, value)
});
}
}
let maskParams = null;
if (number_mask) {
maskParams = handleNumberMask({
mask_options,
number_mask
});
if (isTrue(as_percent)) {
maskParams = handlePercentMask({
props,
locale,
maskParams
});
}
} else if (currency_mask) {
maskParams = handleCurrencyMask({
mask_options,
currency_mask
});
}
return maskParams;
};
//# sourceMappingURL=InputMaskedHooks.js.map