@alfalab/core-components-number-input 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/Component.d.ts +186 -0
- package/Component.js +164 -0
- package/cssm/Component.d.ts +186 -0
- package/cssm/Component.js +164 -0
- package/cssm/index.d.ts +1 -0
- package/cssm/index.js +10 -0
- package/esm/Component.d.ts +186 -0
- package/esm/Component.js +157 -0
- package/esm/index.d.ts +1 -0
- package/esm/index.js +4 -0
- package/index.d.ts +1 -0
- package/index.js +10 -0
- package/modern/Component.d.ts +186 -0
- package/modern/Component.js +116 -0
- package/modern/index.d.ts +1 -0
- package/modern/index.js +4 -0
- package/package.json +23 -0
- package/send-stats.js +82 -0
package/Component.d.ts
ADDED
|
@@ -0,0 +1,186 @@
|
|
|
1
|
+
/// <reference types="react" />
|
|
2
|
+
import React from 'react';
|
|
3
|
+
import { ChangeEvent, InputHTMLAttributes, MouseEvent, ReactNode } from "react";
|
|
4
|
+
type InputProps = Omit<InputHTMLAttributes<HTMLInputElement>, "size" | "type" | "value" | "defaultValue" | "onChange" | "onClick" | "onMouseDown" | "enterKeyHint"> & {
|
|
5
|
+
/**
|
|
6
|
+
* Значение поля ввода
|
|
7
|
+
*/
|
|
8
|
+
value?: string;
|
|
9
|
+
/**
|
|
10
|
+
* Начальное значение поля
|
|
11
|
+
*/
|
|
12
|
+
defaultValue?: string;
|
|
13
|
+
/**
|
|
14
|
+
* Растягивает компонент на ширину контейнера
|
|
15
|
+
*/
|
|
16
|
+
block?: boolean;
|
|
17
|
+
/**
|
|
18
|
+
* Крестик для очистки поля
|
|
19
|
+
*/
|
|
20
|
+
clear?: boolean;
|
|
21
|
+
/**
|
|
22
|
+
* Размер компонента
|
|
23
|
+
*/
|
|
24
|
+
size?: "s" | "m" | "l" | "xl";
|
|
25
|
+
/**
|
|
26
|
+
* Набор цветов для компонента
|
|
27
|
+
*/
|
|
28
|
+
colors?: "default" | "inverted";
|
|
29
|
+
/**
|
|
30
|
+
* Отображение ошибки
|
|
31
|
+
*/
|
|
32
|
+
error?: ReactNode | boolean;
|
|
33
|
+
/**
|
|
34
|
+
* Отображение иконки успеха
|
|
35
|
+
*/
|
|
36
|
+
success?: boolean;
|
|
37
|
+
/**
|
|
38
|
+
* Текст подсказки
|
|
39
|
+
*/
|
|
40
|
+
hint?: ReactNode;
|
|
41
|
+
/**
|
|
42
|
+
* Лейбл компонента
|
|
43
|
+
*/
|
|
44
|
+
label?: React.ReactNode;
|
|
45
|
+
/**
|
|
46
|
+
* Вид лейбла внутри / снаружи
|
|
47
|
+
*/
|
|
48
|
+
labelView?: "inner" | "outer";
|
|
49
|
+
/**
|
|
50
|
+
* Атрибут type
|
|
51
|
+
*/
|
|
52
|
+
type?: "number" | "card" | "email" | "money" | "password" | "tel" | "text";
|
|
53
|
+
/**
|
|
54
|
+
* Ref для обертки input
|
|
55
|
+
*/
|
|
56
|
+
wrapperRef?: React.Ref<HTMLDivElement>;
|
|
57
|
+
/**
|
|
58
|
+
* Слот слева
|
|
59
|
+
*/
|
|
60
|
+
leftAddons?: React.ReactNode;
|
|
61
|
+
/**
|
|
62
|
+
* Слот справа
|
|
63
|
+
*/
|
|
64
|
+
rightAddons?: React.ReactNode;
|
|
65
|
+
/**
|
|
66
|
+
* Слот под инпутом
|
|
67
|
+
*/
|
|
68
|
+
bottomAddons?: React.ReactNode;
|
|
69
|
+
/**
|
|
70
|
+
* Дополнительный класс
|
|
71
|
+
*/
|
|
72
|
+
className?: string;
|
|
73
|
+
/**
|
|
74
|
+
* Дополнительный класс для поля
|
|
75
|
+
*/
|
|
76
|
+
fieldClassName?: string;
|
|
77
|
+
/**
|
|
78
|
+
* Дополнительный класс инпута
|
|
79
|
+
*/
|
|
80
|
+
inputClassName?: string;
|
|
81
|
+
/**
|
|
82
|
+
* Дополнительный класс для лейбла
|
|
83
|
+
*/
|
|
84
|
+
labelClassName?: string;
|
|
85
|
+
/**
|
|
86
|
+
* Дополнительный класс для аддонов
|
|
87
|
+
*/
|
|
88
|
+
addonsClassName?: string;
|
|
89
|
+
/**
|
|
90
|
+
* Класс, который будет установлен при фокусе
|
|
91
|
+
*/
|
|
92
|
+
focusedClassName?: string;
|
|
93
|
+
/**
|
|
94
|
+
* Класс, который будет установлен, если в поле есть значение
|
|
95
|
+
*/
|
|
96
|
+
filledClassName?: string;
|
|
97
|
+
/**
|
|
98
|
+
* Обработчик поля ввода
|
|
99
|
+
*/
|
|
100
|
+
onChange?: (event: ChangeEvent<HTMLInputElement>, payload: {
|
|
101
|
+
value: string;
|
|
102
|
+
}) => void;
|
|
103
|
+
/**
|
|
104
|
+
* Обработчик нажатия на кнопку очистки
|
|
105
|
+
*/
|
|
106
|
+
onClear?: (event: MouseEvent<HTMLButtonElement>) => void;
|
|
107
|
+
/**
|
|
108
|
+
* Обработчик клика по полю
|
|
109
|
+
*/
|
|
110
|
+
onClick?: (event: MouseEvent<HTMLDivElement>) => void;
|
|
111
|
+
/**
|
|
112
|
+
* Обработчик MouseDown по полю
|
|
113
|
+
*/
|
|
114
|
+
onMouseDown?: (event: MouseEvent<HTMLDivElement>) => void;
|
|
115
|
+
/**
|
|
116
|
+
* Обработчик MouseUp по полю
|
|
117
|
+
*/
|
|
118
|
+
onMouseUp?: (event: MouseEvent<HTMLDivElement>) => void;
|
|
119
|
+
/**
|
|
120
|
+
* Идентификатор для систем автоматизированного тестирования
|
|
121
|
+
*/
|
|
122
|
+
dataTestId?: string;
|
|
123
|
+
};
|
|
124
|
+
type NumberInputProps = Omit<InputProps, 'value' | 'onChange' | 'type'> & {
|
|
125
|
+
/**
|
|
126
|
+
* Значение поля ввода
|
|
127
|
+
*/
|
|
128
|
+
value?: string | number | null;
|
|
129
|
+
/**
|
|
130
|
+
* Учитывать знаки '+' и '-'
|
|
131
|
+
*/
|
|
132
|
+
allowSigns?: boolean;
|
|
133
|
+
/**
|
|
134
|
+
* Разделитель ',' или '.'
|
|
135
|
+
*/
|
|
136
|
+
separator?: '.' | ',';
|
|
137
|
+
/**
|
|
138
|
+
* Количество символов после разделителя
|
|
139
|
+
*/
|
|
140
|
+
fractionLength?: number;
|
|
141
|
+
/**
|
|
142
|
+
* Обработчик события изменения значения
|
|
143
|
+
*/
|
|
144
|
+
onChange?: (e: React.ChangeEvent<HTMLInputElement>, payload: {
|
|
145
|
+
/**
|
|
146
|
+
* Числовое значение инпута
|
|
147
|
+
*/
|
|
148
|
+
value: number | null;
|
|
149
|
+
/**
|
|
150
|
+
* Строковое значение инпута
|
|
151
|
+
*/
|
|
152
|
+
valueString: string;
|
|
153
|
+
}) => void;
|
|
154
|
+
};
|
|
155
|
+
declare const NumberInput: React.ForwardRefExoticComponent<Omit<InputProps, "type" | "onChange" | "value"> & {
|
|
156
|
+
/**
|
|
157
|
+
* Значение поля ввода
|
|
158
|
+
*/
|
|
159
|
+
value?: string | number | null | undefined;
|
|
160
|
+
/**
|
|
161
|
+
* Учитывать знаки '+' и '-'
|
|
162
|
+
*/
|
|
163
|
+
allowSigns?: boolean | undefined;
|
|
164
|
+
/**
|
|
165
|
+
* Разделитель ',' или '.'
|
|
166
|
+
*/
|
|
167
|
+
separator?: "." | "," | undefined;
|
|
168
|
+
/**
|
|
169
|
+
* Количество символов после разделителя
|
|
170
|
+
*/
|
|
171
|
+
fractionLength?: number | undefined;
|
|
172
|
+
/**
|
|
173
|
+
* Обработчик события изменения значения
|
|
174
|
+
*/
|
|
175
|
+
onChange?: ((e: React.ChangeEvent<HTMLInputElement>, payload: {
|
|
176
|
+
/**
|
|
177
|
+
* Числовое значение инпута
|
|
178
|
+
*/
|
|
179
|
+
value: number | null;
|
|
180
|
+
/**
|
|
181
|
+
* Строковое значение инпута
|
|
182
|
+
*/
|
|
183
|
+
valueString: string;
|
|
184
|
+
}) => void) | undefined;
|
|
185
|
+
} & React.RefAttributes<HTMLInputElement>>;
|
|
186
|
+
export { NumberInputProps, NumberInput };
|
package/Component.js
ADDED
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var React = require('react');
|
|
4
|
+
var mergeRefs = require('react-merge-refs');
|
|
5
|
+
var coreComponentsMaskedInput = require('@alfalab/core-components-masked-input');
|
|
6
|
+
|
|
7
|
+
function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
|
|
8
|
+
|
|
9
|
+
var React__default = /*#__PURE__*/_interopDefaultCompat(React);
|
|
10
|
+
var mergeRefs__default = /*#__PURE__*/_interopDefaultCompat(mergeRefs);
|
|
11
|
+
|
|
12
|
+
/******************************************************************************
|
|
13
|
+
Copyright (c) Microsoft Corporation.
|
|
14
|
+
|
|
15
|
+
Permission to use, copy, modify, and/or distribute this software for any
|
|
16
|
+
purpose with or without fee is hereby granted.
|
|
17
|
+
|
|
18
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
|
19
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
20
|
+
AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
21
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
22
|
+
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
|
23
|
+
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
24
|
+
PERFORMANCE OF THIS SOFTWARE.
|
|
25
|
+
***************************************************************************** */
|
|
26
|
+
var __assign = function () {
|
|
27
|
+
__assign = Object.assign || function __assign(t) {
|
|
28
|
+
for (var s, i = 1, n = arguments.length; i < n; i++) {
|
|
29
|
+
s = arguments[i];
|
|
30
|
+
for (var p in s)
|
|
31
|
+
if (Object.prototype.hasOwnProperty.call(s, p))
|
|
32
|
+
t[p] = s[p];
|
|
33
|
+
}
|
|
34
|
+
return t;
|
|
35
|
+
};
|
|
36
|
+
return __assign.apply(this, arguments);
|
|
37
|
+
};
|
|
38
|
+
function __rest(s, e) {
|
|
39
|
+
var t = {};
|
|
40
|
+
for (var p in s)
|
|
41
|
+
if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
|
|
42
|
+
t[p] = s[p];
|
|
43
|
+
if (s != null && typeof Object.getOwnPropertySymbols === "function")
|
|
44
|
+
for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
|
|
45
|
+
if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
|
|
46
|
+
t[p[i]] = s[p[i]];
|
|
47
|
+
}
|
|
48
|
+
return t;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
var NumberInput = React__default.default.forwardRef(function (_a, ref) {
|
|
52
|
+
var propValue = _a.value, onChange = _a.onChange, onBlur = _a.onBlur, _b = _a.allowSigns, allowSigns = _b === void 0 ? true : _b, _c = _a.separator, separator = _c === void 0 ? ',' : _c, fractionLength = _a.fractionLength, defaultValue = _a.defaultValue, restProps = __rest(_a, ["value", "onChange", "onBlur", "allowSigns", "separator", "fractionLength", "defaultValue"]);
|
|
53
|
+
var uncontrolled = propValue === undefined;
|
|
54
|
+
var inputRef = React.useRef(null);
|
|
55
|
+
var _d = React.useState(defaultValue || ''), value = _d[0], setValue = _d[1];
|
|
56
|
+
var getNumberValueFromStr = function (valueString) {
|
|
57
|
+
if (valueString.includes(',')) {
|
|
58
|
+
return parseFloat(valueString.replace(',', '.'));
|
|
59
|
+
}
|
|
60
|
+
return parseFloat(valueString);
|
|
61
|
+
};
|
|
62
|
+
var handleChange = function (event, payload) {
|
|
63
|
+
var input = event.target;
|
|
64
|
+
var positionСursor = input.selectionStart || 0;
|
|
65
|
+
if (input.value[positionСursor] === separator) {
|
|
66
|
+
var index = input.selectionStart + 1;
|
|
67
|
+
input.selectionStart = index;
|
|
68
|
+
}
|
|
69
|
+
if (onChange) {
|
|
70
|
+
onChange(event, {
|
|
71
|
+
value: getNumberValueFromStr(input.value),
|
|
72
|
+
valueString: input.value,
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
if (uncontrolled) {
|
|
76
|
+
setValue(payload.value);
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
var formatSeparatorValue = function (separator, rawValue) {
|
|
80
|
+
var singsRegexp = /\W+/;
|
|
81
|
+
var sings = singsRegexp.exec(rawValue);
|
|
82
|
+
if (sings && sings[0] !== ',' && sings[0] !== '.') {
|
|
83
|
+
return rawValue.replace(sings[0], '');
|
|
84
|
+
}
|
|
85
|
+
var separatorRegexp = /\.|,/;
|
|
86
|
+
if (rawValue.length === 1 && separatorRegexp.test(rawValue))
|
|
87
|
+
return '';
|
|
88
|
+
if (separator === ',') {
|
|
89
|
+
return rawValue.replace('.', ',');
|
|
90
|
+
}
|
|
91
|
+
if (separator === '.') {
|
|
92
|
+
return rawValue.replace(',', '.');
|
|
93
|
+
}
|
|
94
|
+
return rawValue;
|
|
95
|
+
};
|
|
96
|
+
var handleBeforeDisplay = function (conformedValue, _a) {
|
|
97
|
+
var rawValue = _a.rawValue, previousConformedValue = _a.previousConformedValue;
|
|
98
|
+
var isTwoSign = rawValue.length - conformedValue.length >= 1;
|
|
99
|
+
var isOneSign = rawValue.length - conformedValue.length === 1;
|
|
100
|
+
var signRegexp = /\.|,/;
|
|
101
|
+
if (fractionLength === 0 && signRegexp.test(rawValue)) {
|
|
102
|
+
return previousConformedValue;
|
|
103
|
+
}
|
|
104
|
+
var zeroRegexp = /^[+|-]?00+$/;
|
|
105
|
+
if (zeroRegexp.test(rawValue)) {
|
|
106
|
+
var index = rawValue.search(/[+|-]/) !== -1 ? 2 : 1;
|
|
107
|
+
return rawValue.slice(0, index);
|
|
108
|
+
}
|
|
109
|
+
if (isTwoSign && previousConformedValue) {
|
|
110
|
+
if (rawValue.length !== conformedValue.length && fractionLength) {
|
|
111
|
+
var valueLength = conformedValue.length - fractionLength;
|
|
112
|
+
var sign = separator === ',' ? '.' : ',';
|
|
113
|
+
if (rawValue.indexOf(sign) !== valueLength &&
|
|
114
|
+
rawValue.indexOf(sign) !== conformedValue.length) {
|
|
115
|
+
return previousConformedValue;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
var hasSign = conformedValue.includes(separator);
|
|
119
|
+
if (!hasSign && isOneSign) {
|
|
120
|
+
return formatSeparatorValue(separator, rawValue);
|
|
121
|
+
}
|
|
122
|
+
return previousConformedValue;
|
|
123
|
+
}
|
|
124
|
+
if (isOneSign && !uncontrolled) {
|
|
125
|
+
return formatSeparatorValue(separator, rawValue);
|
|
126
|
+
}
|
|
127
|
+
return conformedValue;
|
|
128
|
+
};
|
|
129
|
+
var handleBlur = function (event) {
|
|
130
|
+
var valueBlur = event.target.value.replace(new RegExp("\\".concat(separator, "$")), '');
|
|
131
|
+
if (onChange) {
|
|
132
|
+
onChange(event, {
|
|
133
|
+
value: getNumberValueFromStr(valueBlur),
|
|
134
|
+
valueString: valueBlur,
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
if (uncontrolled) {
|
|
138
|
+
setValue(valueBlur);
|
|
139
|
+
}
|
|
140
|
+
if (onBlur)
|
|
141
|
+
onBlur(event);
|
|
142
|
+
};
|
|
143
|
+
var handleMask = function (valueInput) {
|
|
144
|
+
var mask = allowSigns ? [/[+-\d]/] : [/\d/];
|
|
145
|
+
if (valueInput.length <= 1)
|
|
146
|
+
return mask;
|
|
147
|
+
mask.push.apply(mask, Array(valueInput.length - 1).fill(/\d/));
|
|
148
|
+
var dotIndex = valueInput.indexOf(separator);
|
|
149
|
+
var signRegexp = new RegExp("[\\d".concat(separator, "]"));
|
|
150
|
+
var startWithDigit = signRegexp.test(valueInput[0]);
|
|
151
|
+
if (dotIndex > (startWithDigit ? 0 : 1)) {
|
|
152
|
+
mask[dotIndex] = signRegexp;
|
|
153
|
+
}
|
|
154
|
+
if (fractionLength && dotIndex !== -1) {
|
|
155
|
+
var endMaskLength = dotIndex + fractionLength + 1;
|
|
156
|
+
mask = mask.slice(0, endMaskLength);
|
|
157
|
+
}
|
|
158
|
+
return mask;
|
|
159
|
+
};
|
|
160
|
+
var visibleValue = uncontrolled ? value : propValue === null || propValue === void 0 ? void 0 : propValue.toString();
|
|
161
|
+
return (React__default.default.createElement(coreComponentsMaskedInput.MaskedInput, __assign({ ref: mergeRefs__default.default([ref, inputRef]), mask: handleMask, value: visibleValue, onBlur: handleBlur, onChange: handleChange, inputMode: 'decimal', onBeforeDisplay: handleBeforeDisplay }, restProps)));
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
exports.NumberInput = NumberInput;
|
|
@@ -0,0 +1,186 @@
|
|
|
1
|
+
/// <reference types="react" />
|
|
2
|
+
import React from 'react';
|
|
3
|
+
import { ChangeEvent, InputHTMLAttributes, MouseEvent, ReactNode } from "react";
|
|
4
|
+
type InputProps = Omit<InputHTMLAttributes<HTMLInputElement>, "size" | "type" | "value" | "defaultValue" | "onChange" | "onClick" | "onMouseDown" | "enterKeyHint"> & {
|
|
5
|
+
/**
|
|
6
|
+
* Значение поля ввода
|
|
7
|
+
*/
|
|
8
|
+
value?: string;
|
|
9
|
+
/**
|
|
10
|
+
* Начальное значение поля
|
|
11
|
+
*/
|
|
12
|
+
defaultValue?: string;
|
|
13
|
+
/**
|
|
14
|
+
* Растягивает компонент на ширину контейнера
|
|
15
|
+
*/
|
|
16
|
+
block?: boolean;
|
|
17
|
+
/**
|
|
18
|
+
* Крестик для очистки поля
|
|
19
|
+
*/
|
|
20
|
+
clear?: boolean;
|
|
21
|
+
/**
|
|
22
|
+
* Размер компонента
|
|
23
|
+
*/
|
|
24
|
+
size?: "s" | "m" | "l" | "xl";
|
|
25
|
+
/**
|
|
26
|
+
* Набор цветов для компонента
|
|
27
|
+
*/
|
|
28
|
+
colors?: "default" | "inverted";
|
|
29
|
+
/**
|
|
30
|
+
* Отображение ошибки
|
|
31
|
+
*/
|
|
32
|
+
error?: ReactNode | boolean;
|
|
33
|
+
/**
|
|
34
|
+
* Отображение иконки успеха
|
|
35
|
+
*/
|
|
36
|
+
success?: boolean;
|
|
37
|
+
/**
|
|
38
|
+
* Текст подсказки
|
|
39
|
+
*/
|
|
40
|
+
hint?: ReactNode;
|
|
41
|
+
/**
|
|
42
|
+
* Лейбл компонента
|
|
43
|
+
*/
|
|
44
|
+
label?: React.ReactNode;
|
|
45
|
+
/**
|
|
46
|
+
* Вид лейбла внутри / снаружи
|
|
47
|
+
*/
|
|
48
|
+
labelView?: "inner" | "outer";
|
|
49
|
+
/**
|
|
50
|
+
* Атрибут type
|
|
51
|
+
*/
|
|
52
|
+
type?: "number" | "card" | "email" | "money" | "password" | "tel" | "text";
|
|
53
|
+
/**
|
|
54
|
+
* Ref для обертки input
|
|
55
|
+
*/
|
|
56
|
+
wrapperRef?: React.Ref<HTMLDivElement>;
|
|
57
|
+
/**
|
|
58
|
+
* Слот слева
|
|
59
|
+
*/
|
|
60
|
+
leftAddons?: React.ReactNode;
|
|
61
|
+
/**
|
|
62
|
+
* Слот справа
|
|
63
|
+
*/
|
|
64
|
+
rightAddons?: React.ReactNode;
|
|
65
|
+
/**
|
|
66
|
+
* Слот под инпутом
|
|
67
|
+
*/
|
|
68
|
+
bottomAddons?: React.ReactNode;
|
|
69
|
+
/**
|
|
70
|
+
* Дополнительный класс
|
|
71
|
+
*/
|
|
72
|
+
className?: string;
|
|
73
|
+
/**
|
|
74
|
+
* Дополнительный класс для поля
|
|
75
|
+
*/
|
|
76
|
+
fieldClassName?: string;
|
|
77
|
+
/**
|
|
78
|
+
* Дополнительный класс инпута
|
|
79
|
+
*/
|
|
80
|
+
inputClassName?: string;
|
|
81
|
+
/**
|
|
82
|
+
* Дополнительный класс для лейбла
|
|
83
|
+
*/
|
|
84
|
+
labelClassName?: string;
|
|
85
|
+
/**
|
|
86
|
+
* Дополнительный класс для аддонов
|
|
87
|
+
*/
|
|
88
|
+
addonsClassName?: string;
|
|
89
|
+
/**
|
|
90
|
+
* Класс, который будет установлен при фокусе
|
|
91
|
+
*/
|
|
92
|
+
focusedClassName?: string;
|
|
93
|
+
/**
|
|
94
|
+
* Класс, который будет установлен, если в поле есть значение
|
|
95
|
+
*/
|
|
96
|
+
filledClassName?: string;
|
|
97
|
+
/**
|
|
98
|
+
* Обработчик поля ввода
|
|
99
|
+
*/
|
|
100
|
+
onChange?: (event: ChangeEvent<HTMLInputElement>, payload: {
|
|
101
|
+
value: string;
|
|
102
|
+
}) => void;
|
|
103
|
+
/**
|
|
104
|
+
* Обработчик нажатия на кнопку очистки
|
|
105
|
+
*/
|
|
106
|
+
onClear?: (event: MouseEvent<HTMLButtonElement>) => void;
|
|
107
|
+
/**
|
|
108
|
+
* Обработчик клика по полю
|
|
109
|
+
*/
|
|
110
|
+
onClick?: (event: MouseEvent<HTMLDivElement>) => void;
|
|
111
|
+
/**
|
|
112
|
+
* Обработчик MouseDown по полю
|
|
113
|
+
*/
|
|
114
|
+
onMouseDown?: (event: MouseEvent<HTMLDivElement>) => void;
|
|
115
|
+
/**
|
|
116
|
+
* Обработчик MouseUp по полю
|
|
117
|
+
*/
|
|
118
|
+
onMouseUp?: (event: MouseEvent<HTMLDivElement>) => void;
|
|
119
|
+
/**
|
|
120
|
+
* Идентификатор для систем автоматизированного тестирования
|
|
121
|
+
*/
|
|
122
|
+
dataTestId?: string;
|
|
123
|
+
};
|
|
124
|
+
type NumberInputProps = Omit<InputProps, 'value' | 'onChange' | 'type'> & {
|
|
125
|
+
/**
|
|
126
|
+
* Значение поля ввода
|
|
127
|
+
*/
|
|
128
|
+
value?: string | number | null;
|
|
129
|
+
/**
|
|
130
|
+
* Учитывать знаки '+' и '-'
|
|
131
|
+
*/
|
|
132
|
+
allowSigns?: boolean;
|
|
133
|
+
/**
|
|
134
|
+
* Разделитель ',' или '.'
|
|
135
|
+
*/
|
|
136
|
+
separator?: '.' | ',';
|
|
137
|
+
/**
|
|
138
|
+
* Количество символов после разделителя
|
|
139
|
+
*/
|
|
140
|
+
fractionLength?: number;
|
|
141
|
+
/**
|
|
142
|
+
* Обработчик события изменения значения
|
|
143
|
+
*/
|
|
144
|
+
onChange?: (e: React.ChangeEvent<HTMLInputElement>, payload: {
|
|
145
|
+
/**
|
|
146
|
+
* Числовое значение инпута
|
|
147
|
+
*/
|
|
148
|
+
value: number | null;
|
|
149
|
+
/**
|
|
150
|
+
* Строковое значение инпута
|
|
151
|
+
*/
|
|
152
|
+
valueString: string;
|
|
153
|
+
}) => void;
|
|
154
|
+
};
|
|
155
|
+
declare const NumberInput: React.ForwardRefExoticComponent<Omit<InputProps, "type" | "onChange" | "value"> & {
|
|
156
|
+
/**
|
|
157
|
+
* Значение поля ввода
|
|
158
|
+
*/
|
|
159
|
+
value?: string | number | null | undefined;
|
|
160
|
+
/**
|
|
161
|
+
* Учитывать знаки '+' и '-'
|
|
162
|
+
*/
|
|
163
|
+
allowSigns?: boolean | undefined;
|
|
164
|
+
/**
|
|
165
|
+
* Разделитель ',' или '.'
|
|
166
|
+
*/
|
|
167
|
+
separator?: "." | "," | undefined;
|
|
168
|
+
/**
|
|
169
|
+
* Количество символов после разделителя
|
|
170
|
+
*/
|
|
171
|
+
fractionLength?: number | undefined;
|
|
172
|
+
/**
|
|
173
|
+
* Обработчик события изменения значения
|
|
174
|
+
*/
|
|
175
|
+
onChange?: ((e: React.ChangeEvent<HTMLInputElement>, payload: {
|
|
176
|
+
/**
|
|
177
|
+
* Числовое значение инпута
|
|
178
|
+
*/
|
|
179
|
+
value: number | null;
|
|
180
|
+
/**
|
|
181
|
+
* Строковое значение инпута
|
|
182
|
+
*/
|
|
183
|
+
valueString: string;
|
|
184
|
+
}) => void) | undefined;
|
|
185
|
+
} & React.RefAttributes<HTMLInputElement>>;
|
|
186
|
+
export { NumberInputProps, NumberInput };
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var React = require('react');
|
|
4
|
+
var mergeRefs = require('react-merge-refs');
|
|
5
|
+
var coreComponentsMaskedInput = require('@alfalab/core-components-masked-input/cssm');
|
|
6
|
+
|
|
7
|
+
function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
|
|
8
|
+
|
|
9
|
+
var React__default = /*#__PURE__*/_interopDefaultCompat(React);
|
|
10
|
+
var mergeRefs__default = /*#__PURE__*/_interopDefaultCompat(mergeRefs);
|
|
11
|
+
|
|
12
|
+
/******************************************************************************
|
|
13
|
+
Copyright (c) Microsoft Corporation.
|
|
14
|
+
|
|
15
|
+
Permission to use, copy, modify, and/or distribute this software for any
|
|
16
|
+
purpose with or without fee is hereby granted.
|
|
17
|
+
|
|
18
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
|
19
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
20
|
+
AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
21
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
22
|
+
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
|
23
|
+
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
24
|
+
PERFORMANCE OF THIS SOFTWARE.
|
|
25
|
+
***************************************************************************** */
|
|
26
|
+
var __assign = function () {
|
|
27
|
+
__assign = Object.assign || function __assign(t) {
|
|
28
|
+
for (var s, i = 1, n = arguments.length; i < n; i++) {
|
|
29
|
+
s = arguments[i];
|
|
30
|
+
for (var p in s)
|
|
31
|
+
if (Object.prototype.hasOwnProperty.call(s, p))
|
|
32
|
+
t[p] = s[p];
|
|
33
|
+
}
|
|
34
|
+
return t;
|
|
35
|
+
};
|
|
36
|
+
return __assign.apply(this, arguments);
|
|
37
|
+
};
|
|
38
|
+
function __rest(s, e) {
|
|
39
|
+
var t = {};
|
|
40
|
+
for (var p in s)
|
|
41
|
+
if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
|
|
42
|
+
t[p] = s[p];
|
|
43
|
+
if (s != null && typeof Object.getOwnPropertySymbols === "function")
|
|
44
|
+
for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
|
|
45
|
+
if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
|
|
46
|
+
t[p[i]] = s[p[i]];
|
|
47
|
+
}
|
|
48
|
+
return t;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
var NumberInput = React__default.default.forwardRef(function (_a, ref) {
|
|
52
|
+
var propValue = _a.value, onChange = _a.onChange, onBlur = _a.onBlur, _b = _a.allowSigns, allowSigns = _b === void 0 ? true : _b, _c = _a.separator, separator = _c === void 0 ? ',' : _c, fractionLength = _a.fractionLength, defaultValue = _a.defaultValue, restProps = __rest(_a, ["value", "onChange", "onBlur", "allowSigns", "separator", "fractionLength", "defaultValue"]);
|
|
53
|
+
var uncontrolled = propValue === undefined;
|
|
54
|
+
var inputRef = React.useRef(null);
|
|
55
|
+
var _d = React.useState(defaultValue || ''), value = _d[0], setValue = _d[1];
|
|
56
|
+
var getNumberValueFromStr = function (valueString) {
|
|
57
|
+
if (valueString.includes(',')) {
|
|
58
|
+
return parseFloat(valueString.replace(',', '.'));
|
|
59
|
+
}
|
|
60
|
+
return parseFloat(valueString);
|
|
61
|
+
};
|
|
62
|
+
var handleChange = function (event, payload) {
|
|
63
|
+
var input = event.target;
|
|
64
|
+
var positionСursor = input.selectionStart || 0;
|
|
65
|
+
if (input.value[positionСursor] === separator) {
|
|
66
|
+
var index = input.selectionStart + 1;
|
|
67
|
+
input.selectionStart = index;
|
|
68
|
+
}
|
|
69
|
+
if (onChange) {
|
|
70
|
+
onChange(event, {
|
|
71
|
+
value: getNumberValueFromStr(input.value),
|
|
72
|
+
valueString: input.value,
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
if (uncontrolled) {
|
|
76
|
+
setValue(payload.value);
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
var formatSeparatorValue = function (separator, rawValue) {
|
|
80
|
+
var singsRegexp = /\W+/;
|
|
81
|
+
var sings = singsRegexp.exec(rawValue);
|
|
82
|
+
if (sings && sings[0] !== ',' && sings[0] !== '.') {
|
|
83
|
+
return rawValue.replace(sings[0], '');
|
|
84
|
+
}
|
|
85
|
+
var separatorRegexp = /\.|,/;
|
|
86
|
+
if (rawValue.length === 1 && separatorRegexp.test(rawValue))
|
|
87
|
+
return '';
|
|
88
|
+
if (separator === ',') {
|
|
89
|
+
return rawValue.replace('.', ',');
|
|
90
|
+
}
|
|
91
|
+
if (separator === '.') {
|
|
92
|
+
return rawValue.replace(',', '.');
|
|
93
|
+
}
|
|
94
|
+
return rawValue;
|
|
95
|
+
};
|
|
96
|
+
var handleBeforeDisplay = function (conformedValue, _a) {
|
|
97
|
+
var rawValue = _a.rawValue, previousConformedValue = _a.previousConformedValue;
|
|
98
|
+
var isTwoSign = rawValue.length - conformedValue.length >= 1;
|
|
99
|
+
var isOneSign = rawValue.length - conformedValue.length === 1;
|
|
100
|
+
var signRegexp = /\.|,/;
|
|
101
|
+
if (fractionLength === 0 && signRegexp.test(rawValue)) {
|
|
102
|
+
return previousConformedValue;
|
|
103
|
+
}
|
|
104
|
+
var zeroRegexp = /^[+|-]?00+$/;
|
|
105
|
+
if (zeroRegexp.test(rawValue)) {
|
|
106
|
+
var index = rawValue.search(/[+|-]/) !== -1 ? 2 : 1;
|
|
107
|
+
return rawValue.slice(0, index);
|
|
108
|
+
}
|
|
109
|
+
if (isTwoSign && previousConformedValue) {
|
|
110
|
+
if (rawValue.length !== conformedValue.length && fractionLength) {
|
|
111
|
+
var valueLength = conformedValue.length - fractionLength;
|
|
112
|
+
var sign = separator === ',' ? '.' : ',';
|
|
113
|
+
if (rawValue.indexOf(sign) !== valueLength &&
|
|
114
|
+
rawValue.indexOf(sign) !== conformedValue.length) {
|
|
115
|
+
return previousConformedValue;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
var hasSign = conformedValue.includes(separator);
|
|
119
|
+
if (!hasSign && isOneSign) {
|
|
120
|
+
return formatSeparatorValue(separator, rawValue);
|
|
121
|
+
}
|
|
122
|
+
return previousConformedValue;
|
|
123
|
+
}
|
|
124
|
+
if (isOneSign && !uncontrolled) {
|
|
125
|
+
return formatSeparatorValue(separator, rawValue);
|
|
126
|
+
}
|
|
127
|
+
return conformedValue;
|
|
128
|
+
};
|
|
129
|
+
var handleBlur = function (event) {
|
|
130
|
+
var valueBlur = event.target.value.replace(new RegExp("\\".concat(separator, "$")), '');
|
|
131
|
+
if (onChange) {
|
|
132
|
+
onChange(event, {
|
|
133
|
+
value: getNumberValueFromStr(valueBlur),
|
|
134
|
+
valueString: valueBlur,
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
if (uncontrolled) {
|
|
138
|
+
setValue(valueBlur);
|
|
139
|
+
}
|
|
140
|
+
if (onBlur)
|
|
141
|
+
onBlur(event);
|
|
142
|
+
};
|
|
143
|
+
var handleMask = function (valueInput) {
|
|
144
|
+
var mask = allowSigns ? [/[+-\d]/] : [/\d/];
|
|
145
|
+
if (valueInput.length <= 1)
|
|
146
|
+
return mask;
|
|
147
|
+
mask.push.apply(mask, Array(valueInput.length - 1).fill(/\d/));
|
|
148
|
+
var dotIndex = valueInput.indexOf(separator);
|
|
149
|
+
var signRegexp = new RegExp("[\\d".concat(separator, "]"));
|
|
150
|
+
var startWithDigit = signRegexp.test(valueInput[0]);
|
|
151
|
+
if (dotIndex > (startWithDigit ? 0 : 1)) {
|
|
152
|
+
mask[dotIndex] = signRegexp;
|
|
153
|
+
}
|
|
154
|
+
if (fractionLength && dotIndex !== -1) {
|
|
155
|
+
var endMaskLength = dotIndex + fractionLength + 1;
|
|
156
|
+
mask = mask.slice(0, endMaskLength);
|
|
157
|
+
}
|
|
158
|
+
return mask;
|
|
159
|
+
};
|
|
160
|
+
var visibleValue = uncontrolled ? value : propValue === null || propValue === void 0 ? void 0 : propValue.toString();
|
|
161
|
+
return (React__default.default.createElement(coreComponentsMaskedInput.MaskedInput, __assign({ ref: mergeRefs__default.default([ref, inputRef]), mask: handleMask, value: visibleValue, onBlur: handleBlur, onChange: handleChange, inputMode: 'decimal', onBeforeDisplay: handleBeforeDisplay }, restProps)));
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
exports.NumberInput = NumberInput;
|