@sproutsocial/seeds-react-token-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.
@@ -0,0 +1,104 @@
1
+ import * as react_jsx_runtime from 'react/jsx-runtime';
2
+ import * as React from 'react';
3
+ import { TypeStyledComponentsCommonProps, TypeSystemCommonProps } from '@sproutsocial/seeds-react-system-props';
4
+ import { TypeIconName, TypeIconProps } from '@sproutsocial/seeds-react-icon';
5
+
6
+ interface TypeTokenSpec {
7
+ id: string;
8
+ iconName?: TypeIconName;
9
+ iconProps?: Omit<TypeIconProps, "name">;
10
+ value: string;
11
+ valid?: boolean;
12
+ }
13
+ interface TypeBaseTokenInputProps {
14
+ /** ID of the form element, should match the "for" value of the associated label */
15
+ id: string;
16
+ name: string;
17
+ iconName?: TypeIconName;
18
+ /** Array of delimiter key names */
19
+ delimiters?: string[];
20
+ /** Current input text. Required when controlling the input text */
21
+ value?: string;
22
+ /** Current focus state. Required when controlling the focus via onFocus and onBlur */
23
+ hasFocus?: boolean;
24
+ /** Id of the currently selected token */
25
+ activeToken?: string;
26
+ /** Array of current tokens */
27
+ tokens?: TypeTokenSpec[];
28
+ /** Standard control of changing tokens. For fine-grain control use onAddToken and onRemoveToken, instead */
29
+ onChangeTokens?: (tokens: TypeTokenSpec[]) => void;
30
+ /** Fine-grained control of adding tokens. Use with onRemoveToken instead of onChangeTokens */
31
+ onAddToken?: (tokenSpec: TypeTokenSpec) => void;
32
+ /** Fine-grained control of removing tokens. Use with onAddToken instead of onChangeTokens */
33
+ onRemoveToken?: (tokenId: string) => void;
34
+ /** Controls clicking on a token. When absent, clicking a token removes itself */
35
+ onClickToken?: (e: React.SyntheticEvent<HTMLButtonElement>, tokenId: string) => void;
36
+ /** Fine-grained control of the input text used to create tokens */
37
+ onChange?: (e: React.SyntheticEvent<HTMLInputElement>, value: string) => void;
38
+ /** Fine-grained control of pasted text */
39
+ onPaste?: (e: React.ClipboardEvent<HTMLInputElement>, value: string) => void;
40
+ onBlur?: (e: React.FocusEvent<HTMLInputElement>) => void;
41
+ onFocus?: (e: React.FocusEvent<HTMLInputElement>) => void;
42
+ onKeyDown?: (e: React.KeyboardEvent<HTMLInputElement>, value: string) => void;
43
+ onKeyUp?: (e: React.KeyboardEvent<HTMLInputElement>, value: string) => void;
44
+ /** Attribute used to associate other elements that describe the Input, like an error */
45
+ ariaDescribedby?: string;
46
+ /** Label used to describe the input if not used with an accompanying visual label */
47
+ ariaLabel?: string;
48
+ /** Placeholder text for when value is undefined or empty */
49
+ placeholder?: string;
50
+ /** Will autofocus the element when mounted to the DOM */
51
+ autoFocus?: boolean;
52
+ /** HTML disabled attribute */
53
+ disabled?: boolean;
54
+ /** Whether or not the current contents of the input are invalid */
55
+ isInvalid?: boolean;
56
+ /** Whether or not the current contents of the input has any warnings */
57
+ hasWarning?: boolean;
58
+ /** HTML required attribute */
59
+ required?: boolean;
60
+ /** 16x16 element, such as an icon */
61
+ elemBefore?: React.ReactNode;
62
+ /** 16x16 element, such as an icon */
63
+ elemAfter?: React.ReactNode;
64
+ /** Max input text length */
65
+ maxLength?: number;
66
+ /** Max length of the token */
67
+ tokenMaxLength?: number;
68
+ /** Props to spread onto the underlying input element */
69
+ inputProps?: React.ComponentPropsWithoutRef<"input">;
70
+ /** Used to get a reference to the underlying element */
71
+ innerRef?: React.Ref<HTMLInputElement>;
72
+ qa?: object;
73
+ /** Browser autocomplete support */
74
+ autocomplete?: string;
75
+ }
76
+ interface TypeTokenInputProps extends TypeBaseTokenInputProps, TypeStyledComponentsCommonProps, TypeSystemCommonProps, Omit<React.ComponentPropsWithoutRef<"div">, keyof TypeBaseTokenInputProps | "color"> {
77
+ }
78
+
79
+ type TypeState = {
80
+ prevProps: TypeTokenInputProps;
81
+ hasFocus: boolean;
82
+ activeToken: string | null | undefined;
83
+ value: string;
84
+ };
85
+ declare class TokenInput extends React.Component<TypeTokenInputProps, TypeState> {
86
+ delimiterMatcher: RegExp;
87
+ constructor(props: TypeTokenInputProps);
88
+ static getDerivedStateFromProps(props: Readonly<TypeTokenInputProps>, state: TypeState): Partial<TypeState>;
89
+ isDelimiter(keyName: string): boolean;
90
+ spawnNewTokens(texts: string[]): void;
91
+ deleteToken(tokenId?: string): void;
92
+ handleChangeText: (e: React.SyntheticEvent<HTMLInputElement>) => void;
93
+ handleFocus: (e: React.FocusEvent<HTMLInputElement>) => void;
94
+ handleBlur: (e: React.FocusEvent<HTMLInputElement>) => void;
95
+ handleKeyUp: (e: React.KeyboardEvent<HTMLInputElement>) => void;
96
+ handleKeyDown: (e: React.KeyboardEvent<HTMLInputElement>) => void;
97
+ handlePaste: (e: React.ClipboardEvent<HTMLInputElement>) => void;
98
+ handleClickToken: (e: React.SyntheticEvent<HTMLButtonElement>, token: TypeTokenSpec) => void;
99
+ renderToken(token: TypeTokenSpec): React.ReactNode;
100
+ renderTokens(tokens: TypeTokenSpec[]): React.ReactNode;
101
+ render(): react_jsx_runtime.JSX.Element;
102
+ }
103
+
104
+ export { TokenInput, type TypeBaseTokenInputProps, type TypeTokenInputProps, type TypeTokenSpec, TokenInput as default };
package/dist/index.js ADDED
@@ -0,0 +1,478 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __export = (target, all) => {
9
+ for (var name in all)
10
+ __defProp(target, name, { get: all[name], enumerable: true });
11
+ };
12
+ var __copyProps = (to, from, except, desc) => {
13
+ if (from && typeof from === "object" || typeof from === "function") {
14
+ for (let key of __getOwnPropNames(from))
15
+ if (!__hasOwnProp.call(to, key) && key !== except)
16
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
17
+ }
18
+ return to;
19
+ };
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ // If the importer is in node compatibility mode or this is not an ESM
22
+ // file that has been converted to a CommonJS file using a Babel-
23
+ // compatible transform (i.e. "__esModule" has not been set), then set
24
+ // "default" to the CommonJS "module.exports" for node compatibility.
25
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
+ mod
27
+ ));
28
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
29
+
30
+ // src/index.ts
31
+ var src_exports = {};
32
+ __export(src_exports, {
33
+ TokenInput: () => TokenInput,
34
+ default: () => src_default
35
+ });
36
+ module.exports = __toCommonJS(src_exports);
37
+
38
+ // src/TokenInput.tsx
39
+ var React2 = __toESM(require("react"));
40
+ var import_seeds_react_input = require("@sproutsocial/seeds-react-input");
41
+ var import_seeds_react_box = __toESM(require("@sproutsocial/seeds-react-box"));
42
+ var import_seeds_react_icon = __toESM(require("@sproutsocial/seeds-react-icon"));
43
+ var import_seeds_react_token = __toESM(require("@sproutsocial/seeds-react-token"));
44
+
45
+ // src/styles.ts
46
+ var import_styled_components = __toESM(require("styled-components"));
47
+ var import_seeds_react_system_props = require("@sproutsocial/seeds-react-system-props");
48
+ var import_seeds_react_mixins = require("@sproutsocial/seeds-react-mixins");
49
+ var Container = import_styled_components.default.div`
50
+ box-sizing: border-box;
51
+ position: relative;
52
+ display: flex;
53
+ flex-wrap: wrap;
54
+ align-items: center;
55
+ align-content: center;
56
+ cursor: text;
57
+ width: 100%;
58
+ border: 1px solid ${(props) => props.theme.colors.form.border.base};
59
+ border-radius: ${(props) => props.theme.radii[500]};
60
+ margin: 0;
61
+ padding: ${(props) => props.theme.space[300]};
62
+ padding-top: ${(props) => props.theme.space[200]};
63
+ background-color: ${(props) => props.theme.colors.form.background.base};
64
+ color: ${(props) => props.theme.colors.text.body};
65
+ transition: border-color ${(props) => props.theme.duration.fast}
66
+ ${(props) => props.theme.easing.ease_in},
67
+ box-shadow ${(props) => props.theme.duration.fast}
68
+ ${(props) => props.theme.easing.ease_in};
69
+ ${(props) => props.theme.typography[200]};
70
+ font-family: ${(props) => props.theme.fontFamily};
71
+ font-weight: ${(props) => props.theme.fontWeights.normal};
72
+ appearance: none;
73
+
74
+ button {
75
+ margin: ${(props) => props.theme.space[200]}
76
+ ${(props) => props.theme.space[200]} 0 0;
77
+ }
78
+
79
+ input {
80
+ ${(props) => props.theme.typography[200]};
81
+ outline: none;
82
+ border: none;
83
+ flex: 1;
84
+ padding: 0;
85
+ padding-top: ${(props) => props.theme.space[100]};
86
+ margin: ${(props) => props.theme.space[200]}
87
+ ${(props) => props.theme.space[300]} ${(props) => props.theme.space[100]}
88
+ 0;
89
+ color: ${(props) => props.theme.colors.text.body};
90
+ background-color: ${(props) => props.theme.colors.form.background.base};
91
+ /** This matches the height of the token so size does not change as tokens are added */
92
+ min-height: 20px;
93
+
94
+ &::-webkit-search-cancel-button {
95
+ appearance: none;
96
+ }
97
+
98
+ /* Explicitly removes double focus ring in environments where box-shadow focus styles have been specified (Seeds). Focus is passed up from the input to the parent container. */
99
+ &:focus {
100
+ box-shadow: none;
101
+ }
102
+
103
+ /* https://stackoverflow.com/questions/14007655/remove-ie10s-clear-field-x-button-on-certain-inputs */
104
+ &::-ms-clear {
105
+ display: none;
106
+ }
107
+
108
+ /* Fix for red ring when input is marked required in Firefox */
109
+ &:not(output):not(:focus):-moz-ui-invalid {
110
+ box-shadow: none;
111
+ }
112
+
113
+ &::placeholder {
114
+ color: ${(props) => props.theme.colors.form.placeholder.base};
115
+ font-style: italic;
116
+ }
117
+
118
+ ${(props) => props.disabled && import_styled_components.css`
119
+ opacity: 0.4;
120
+
121
+ cursor: not-allowed;
122
+ `}
123
+ }
124
+
125
+ ${(props) => props.hasBeforeElement && import_styled_components.css`
126
+ padding-left: 40px;
127
+ `}
128
+
129
+ ${(props) => props.hasAfterElement && import_styled_components.css`
130
+ padding-right: 40px;
131
+ `}
132
+
133
+
134
+ ${(props) => props.disabled && import_styled_components.css`
135
+ opacity: 0.4;
136
+
137
+ cursor: not-allowed;
138
+ `}
139
+
140
+ ${(props) => props.focused && import_styled_components.css`
141
+ ${import_seeds_react_mixins.focusRing}
142
+ `}
143
+
144
+ ${(props) => props.invalid && import_styled_components.css`
145
+ border-color: ${(props2) => props2.theme.colors.form.border.error};
146
+ `}
147
+
148
+ ${(props) => props.warning && import_styled_components.css`
149
+ border-color: ${(props2) => props2.theme.colors.form.border.warning};
150
+ `}
151
+
152
+ ${import_seeds_react_system_props.COMMON}
153
+ `;
154
+ Container.displayName = "TokenInputContainer";
155
+ var styles_default = Container;
156
+
157
+ // src/util.ts
158
+ var import_lodash = __toESM(require("lodash.uniqueid"));
159
+ var asTokenSpec = (text) => ({
160
+ id: (0, import_lodash.default)(`${text}-`),
161
+ value: text.trim()
162
+ });
163
+ var KeyNameToRegExpChar = {
164
+ Enter: "\\n"
165
+ };
166
+ var delimitersAsRegExp = (delimiters) => {
167
+ if (!delimiters)
168
+ return /[,\n]/;
169
+ const chars = delimiters.map(
170
+ (key) => KeyNameToRegExpChar[key] || key
171
+ ).join("");
172
+ return RegExp(`[${chars}]`);
173
+ };
174
+
175
+ // src/TokenScreenReaderStatus.tsx
176
+ var import_react = require("react");
177
+ var import_react2 = require("react");
178
+
179
+ // ../seeds-react-visually-hidden/dist/esm/index.js
180
+ var import_styled_components2 = __toESM(require("styled-components"));
181
+ var import_seeds_react_mixins2 = require("@sproutsocial/seeds-react-mixins");
182
+ var import_jsx_runtime = require("react/jsx-runtime");
183
+ var StyledVisuallyHidden = import_styled_components2.default.span`
184
+ ${import_seeds_react_mixins2.visuallyHidden}
185
+ `;
186
+ var VisuallyHidden = (props) => {
187
+ return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(StyledVisuallyHidden, { ...props });
188
+ };
189
+
190
+ // src/TokenScreenReaderStatus.tsx
191
+ var import_jsx_runtime2 = require("react/jsx-runtime");
192
+ function usePrevious(value) {
193
+ const ref = (0, import_react2.useRef)();
194
+ (0, import_react2.useEffect)(() => {
195
+ ref.current = value;
196
+ });
197
+ return ref.current;
198
+ }
199
+ var TokenScreenReaderStatus = ({
200
+ tokens
201
+ }) => {
202
+ const prevTokens = usePrevious(tokens);
203
+ const [tokenStatus, setTokenStatus] = (0, import_react2.useState)("");
204
+ (0, import_react2.useEffect)(() => {
205
+ if (prevTokens && tokens) {
206
+ if (prevTokens.length > tokens.length) {
207
+ setTokenStatus(
208
+ `${prevTokens.filter((item) => tokens.indexOf(item) === -1)[0]?.value} has been removed`
209
+ );
210
+ }
211
+ if (prevTokens.length < tokens.length) {
212
+ setTokenStatus(`${tokens[tokens.length - 1]?.value} has been added.`);
213
+ }
214
+ }
215
+ }, [prevTokens, tokens, tokenStatus]);
216
+ return /* @__PURE__ */ (0, import_jsx_runtime2.jsx)(VisuallyHidden, { as: "div", role: "status", children: tokenStatus });
217
+ };
218
+
219
+ // src/TokenInput.tsx
220
+ var import_jsx_runtime3 = require("react/jsx-runtime");
221
+ var DefaultDelimiters = [",", "Enter"];
222
+ var ControlledPropNames = [
223
+ "value",
224
+ "hasFocus",
225
+ "activeToken"
226
+ ];
227
+ var TokenInput = class extends React2.Component {
228
+ delimiterMatcher;
229
+ constructor(props) {
230
+ super(props);
231
+ const { hasFocus, activeToken, value, delimiters } = props;
232
+ this.delimiterMatcher = delimitersAsRegExp(delimiters || DefaultDelimiters);
233
+ this.state = {
234
+ prevProps: props,
235
+ hasFocus: hasFocus || false,
236
+ activeToken,
237
+ value: value || ""
238
+ };
239
+ }
240
+ static getDerivedStateFromProps(props, state) {
241
+ const { prevProps } = state;
242
+ const modifiedState = { prevProps: props };
243
+ ControlledPropNames.forEach((propName) => {
244
+ const currentProp = props[propName];
245
+ if (currentProp !== prevProps[propName]) {
246
+ modifiedState[propName] = currentProp;
247
+ }
248
+ });
249
+ modifiedState.prevProps = props;
250
+ return modifiedState;
251
+ }
252
+ isDelimiter(keyName) {
253
+ const { delimiters = DefaultDelimiters } = this.props;
254
+ return delimiters.includes(keyName);
255
+ }
256
+ spawnNewTokens(texts) {
257
+ const {
258
+ onAddToken,
259
+ onChangeTokens,
260
+ tokenMaxLength: max = Infinity,
261
+ tokens = []
262
+ } = this.props;
263
+ const tokenSpecs = texts.map((text) => asTokenSpec(text.slice(0, max)));
264
+ if (onAddToken) {
265
+ tokenSpecs.forEach(onAddToken);
266
+ } else if (onChangeTokens) {
267
+ onChangeTokens(tokens.concat(tokenSpecs));
268
+ }
269
+ this.setState({
270
+ value: ""
271
+ });
272
+ }
273
+ deleteToken(tokenId) {
274
+ const { onRemoveToken, onChangeTokens, tokens = [] } = this.props;
275
+ const count = tokens.length;
276
+ if (count === 0)
277
+ return;
278
+ const id = tokenId ?? tokens[count - 1]?.id;
279
+ if (onRemoveToken) {
280
+ onRemoveToken(id ? id : "");
281
+ } else if (onChangeTokens) {
282
+ onChangeTokens(tokens.filter((tokenSpec) => tokenSpec.id !== id));
283
+ }
284
+ this.setState({
285
+ value: ""
286
+ });
287
+ }
288
+ handleChangeText = (e) => {
289
+ const { value } = e.currentTarget;
290
+ const { onChange } = this.props;
291
+ this.setState({
292
+ value
293
+ });
294
+ onChange?.(e, value);
295
+ };
296
+ handleFocus = (e) => {
297
+ const { onFocus } = this.props;
298
+ this.setState({
299
+ hasFocus: true
300
+ });
301
+ onFocus?.(e);
302
+ };
303
+ handleBlur = (e) => {
304
+ const { onBlur } = this.props;
305
+ if (onBlur)
306
+ onBlur(e);
307
+ this.setState({
308
+ hasFocus: false
309
+ });
310
+ };
311
+ handleKeyUp = (e) => {
312
+ this.props.onKeyUp?.(e, e.currentTarget.value);
313
+ };
314
+ handleKeyDown = (e) => {
315
+ const { onKeyDown } = this.props;
316
+ const { key, currentTarget } = e;
317
+ const text = currentTarget.value;
318
+ if (onKeyDown)
319
+ onKeyDown(e, text);
320
+ if (this.isDelimiter(key)) {
321
+ if (text) {
322
+ this.spawnNewTokens([text]);
323
+ e.preventDefault();
324
+ }
325
+ } else if (key === "Backspace") {
326
+ if (text === "") {
327
+ this.deleteToken();
328
+ }
329
+ }
330
+ };
331
+ handlePaste = (e) => {
332
+ const text = e.clipboardData.getData("text");
333
+ const { onPaste } = this.props;
334
+ if (onPaste)
335
+ onPaste(e, text);
336
+ const subtexts = text.split(this.delimiterMatcher);
337
+ const texts = subtexts.filter((subtext) => subtext.length);
338
+ if (texts.length > 1) {
339
+ this.spawnNewTokens(texts);
340
+ e.preventDefault();
341
+ }
342
+ };
343
+ handleClickToken = (e, token) => {
344
+ const { onClickToken, disabled } = this.props;
345
+ if (onClickToken)
346
+ onClickToken(e, token.id);
347
+ if (!disabled) {
348
+ this.deleteToken(token.id);
349
+ }
350
+ };
351
+ renderToken(token) {
352
+ const { iconName: defaultIconName, disabled } = this.props;
353
+ const activeId = this.state.activeToken;
354
+ const {
355
+ id,
356
+ iconName: tokenIconName,
357
+ iconProps = { "aria-hidden": true },
358
+ value,
359
+ valid
360
+ } = token;
361
+ const iconName = tokenIconName || defaultIconName;
362
+ const isActive = activeId === id;
363
+ return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
364
+ import_seeds_react_token.default,
365
+ {
366
+ id,
367
+ onClick: (e) => this.handleClickToken(e, token),
368
+ valid,
369
+ active: isActive,
370
+ disabled,
371
+ children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_seeds_react_box.default, { display: "flex", alignItems: "center", children: [
372
+ iconName && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(import_seeds_react_icon.default, { name: iconName, size: "mini", pr: 300, ...iconProps }),
373
+ value
374
+ ] })
375
+ }
376
+ );
377
+ }
378
+ renderTokens(tokens) {
379
+ return tokens.map((token) => /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { className: "TokenInput-token", children: this.renderToken(token) }, token.id));
380
+ }
381
+ render() {
382
+ const {
383
+ autoFocus,
384
+ autocomplete,
385
+ disabled,
386
+ isInvalid,
387
+ hasWarning,
388
+ id,
389
+ name,
390
+ placeholder,
391
+ required,
392
+ elemBefore,
393
+ elemAfter,
394
+ maxLength,
395
+ ariaDescribedby,
396
+ ariaLabel,
397
+ innerRef,
398
+ // These functions are used in the class functions above, but need to be extracted in order for `rest` to be correct
399
+ /* eslint-disable @typescript-eslint/no-unused-vars */
400
+ value,
401
+ onAddToken,
402
+ onRemoveToken,
403
+ onChangeTokens,
404
+ onClickToken,
405
+ onBlur,
406
+ onChange,
407
+ onFocus,
408
+ onKeyDown,
409
+ onKeyUp,
410
+ onPaste,
411
+ /* eslint-enable @typescript-eslint/no-unused-vars */
412
+ inputProps = {},
413
+ qa = {},
414
+ tokens,
415
+ ...rest
416
+ } = this.props;
417
+ const { state } = this;
418
+ return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
419
+ styles_default,
420
+ {
421
+ hasBeforeElement: !!elemBefore,
422
+ hasAfterElement: !!elemAfter,
423
+ disabled,
424
+ invalid: !!isInvalid,
425
+ warning: hasWarning,
426
+ focused: state.hasFocus,
427
+ ...rest,
428
+ children: [
429
+ elemBefore && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(import_seeds_react_input.Accessory, { before: true, children: elemBefore }),
430
+ tokens && this.renderTokens(tokens),
431
+ /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(TokenScreenReaderStatus, { tokens }),
432
+ /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
433
+ "input",
434
+ {
435
+ "aria-describedby": ariaDescribedby,
436
+ "aria-invalid": !!isInvalid,
437
+ "aria-label": ariaLabel,
438
+ autoFocus,
439
+ autoComplete: autocomplete,
440
+ disabled,
441
+ id,
442
+ name,
443
+ placeholder,
444
+ type: "text",
445
+ required,
446
+ value: state.value,
447
+ maxLength,
448
+ onBlur: this.handleBlur,
449
+ onChange: this.handleChangeText,
450
+ onFocus: this.handleFocus,
451
+ onKeyDown: this.handleKeyDown,
452
+ onKeyUp: this.handleKeyUp,
453
+ onPaste: this.handlePaste,
454
+ ref: innerRef,
455
+ "data-qa-input": name || "",
456
+ "data-qa-input-isdisabled": !!disabled,
457
+ "data-qa-input-isrequired": !!required,
458
+ ...qa,
459
+ ...inputProps
460
+ }
461
+ ),
462
+ elemAfter && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(import_seeds_react_input.Accessory, { after: true, children: elemAfter })
463
+ ]
464
+ }
465
+ );
466
+ }
467
+ };
468
+
469
+ // src/TokenInputTypes.ts
470
+ var React3 = require("react");
471
+
472
+ // src/index.ts
473
+ var src_default = TokenInput;
474
+ // Annotate the CommonJS export names for ESM import in node:
475
+ 0 && (module.exports = {
476
+ TokenInput
477
+ });
478
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/index.ts","../src/TokenInput.tsx","../src/styles.ts","../src/util.ts","../src/TokenScreenReaderStatus.tsx","../../seeds-react-visually-hidden/src/VisuallyHidden.tsx","../src/TokenInputTypes.ts"],"sourcesContent":["import TokenInput from \"./TokenInput\";\n\nexport default TokenInput;\nexport { TokenInput };\nexport * from \"./TokenInputTypes\";\n","import * as React from \"react\";\nimport { Accessory } from \"@sproutsocial/seeds-react-input\";\nimport Box from \"@sproutsocial/seeds-react-box\";\nimport Icon from \"@sproutsocial/seeds-react-icon\";\nimport Token from \"@sproutsocial/seeds-react-token\";\nimport Container from \"./styles\";\nimport { asTokenSpec, delimitersAsRegExp } from \"./util\";\nimport type { TypeTokenInputProps, TypeTokenSpec } from \"./TokenInputTypes\";\nimport { TokenScreenReaderStatus } from \"./TokenScreenReaderStatus\";\n\ntype TypeState = {\n prevProps: TypeTokenInputProps;\n hasFocus: boolean;\n activeToken: string | null | undefined;\n value: string;\n};\n\nconst DefaultDelimiters = [\",\", \"Enter\"];\nconst ControlledPropNames: (keyof TypeState)[] = [\n \"value\",\n \"hasFocus\",\n \"activeToken\",\n];\n\nexport default class TokenInput extends React.Component<\n TypeTokenInputProps,\n TypeState\n> {\n delimiterMatcher: RegExp;\n\n constructor(props: TypeTokenInputProps) {\n super(props);\n const { hasFocus, activeToken, value, delimiters } = props;\n this.delimiterMatcher = delimitersAsRegExp(delimiters || DefaultDelimiters);\n this.state = {\n prevProps: props,\n hasFocus: hasFocus || false,\n activeToken: activeToken,\n value: value || \"\",\n };\n }\n\n static getDerivedStateFromProps(\n props: Readonly<TypeTokenInputProps>,\n state: TypeState\n ) {\n const { prevProps } = state;\n const modifiedState: Partial<TypeState> = { prevProps: props };\n ControlledPropNames.forEach((propName) => {\n const currentProp = props[propName as keyof TypeTokenInputProps];\n\n // @ts-ignore: TODO - fix state types for prevProps\n if (currentProp !== prevProps[propName]) {\n modifiedState[propName] = currentProp;\n }\n });\n modifiedState.prevProps = props;\n return modifiedState;\n }\n\n isDelimiter(keyName: string) {\n const { delimiters = DefaultDelimiters } = this.props;\n return delimiters.includes(keyName);\n }\n\n spawnNewTokens(texts: string[]) {\n const {\n onAddToken,\n onChangeTokens,\n tokenMaxLength: max = Infinity,\n tokens = [],\n } = this.props;\n const tokenSpecs = texts.map((text) => asTokenSpec(text.slice(0, max)));\n\n if (onAddToken) {\n tokenSpecs.forEach(onAddToken);\n } else if (onChangeTokens) {\n onChangeTokens(tokens.concat(tokenSpecs));\n }\n\n this.setState({\n value: \"\",\n });\n }\n\n deleteToken(tokenId?: string) {\n const { onRemoveToken, onChangeTokens, tokens = [] } = this.props;\n const count = tokens.length;\n if (count === 0) return;\n const id = tokenId ?? tokens[count - 1]?.id;\n\n if (onRemoveToken) {\n onRemoveToken(id ? id : \"\");\n } else if (onChangeTokens) {\n onChangeTokens(tokens.filter((tokenSpec) => tokenSpec.id !== id));\n }\n\n this.setState({\n value: \"\",\n });\n }\n\n handleChangeText = (e: React.SyntheticEvent<HTMLInputElement>) => {\n const { value } = e.currentTarget;\n const { onChange } = this.props;\n this.setState({\n value,\n });\n onChange?.(e, value);\n };\n\n handleFocus = (e: React.FocusEvent<HTMLInputElement>) => {\n const { onFocus } = this.props;\n this.setState({\n hasFocus: true,\n });\n onFocus?.(e);\n };\n\n handleBlur = (e: React.FocusEvent<HTMLInputElement>) => {\n const { onBlur } = this.props;\n if (onBlur) onBlur(e);\n this.setState({\n hasFocus: false,\n });\n };\n\n handleKeyUp = (e: React.KeyboardEvent<HTMLInputElement>) => {\n this.props.onKeyUp?.(e, e.currentTarget.value);\n };\n\n handleKeyDown = (e: React.KeyboardEvent<HTMLInputElement>) => {\n const { onKeyDown } = this.props;\n const { key, currentTarget } = e;\n const text = currentTarget.value;\n if (onKeyDown) onKeyDown(e, text);\n\n // keyPress event runs before change\n // Prevent event from bubbling up and calling change, which can lead to comma in value\n if (this.isDelimiter(key)) {\n if (text) {\n this.spawnNewTokens([text]);\n e.preventDefault();\n }\n } else if (key === \"Backspace\") {\n if (text === \"\") {\n this.deleteToken();\n }\n }\n };\n\n handlePaste = (e: React.ClipboardEvent<HTMLInputElement>) => {\n const text = e.clipboardData.getData(\"text\");\n const { onPaste } = this.props;\n if (onPaste) onPaste(e, text);\n const subtexts = text.split(this.delimiterMatcher);\n const texts = subtexts.filter((subtext) => subtext.length);\n\n if (texts.length > 1) {\n this.spawnNewTokens(texts);\n e.preventDefault();\n }\n };\n\n handleClickToken = (\n e: React.SyntheticEvent<HTMLButtonElement>,\n token: TypeTokenSpec\n ) => {\n const { onClickToken, disabled } = this.props;\n if (onClickToken) onClickToken(e, token.id);\n\n if (!disabled) {\n this.deleteToken(token.id);\n }\n };\n\n renderToken(token: TypeTokenSpec): React.ReactNode {\n const { iconName: defaultIconName, disabled } = this.props;\n const activeId = this.state.activeToken;\n const {\n id,\n iconName: tokenIconName,\n iconProps = { \"aria-hidden\": true },\n value,\n valid,\n } = token;\n const iconName = tokenIconName || defaultIconName;\n const isActive = activeId === id;\n return (\n <Token\n id={id}\n onClick={(e) => this.handleClickToken(e, token)}\n valid={valid}\n // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n // @ts-ignore\n active={isActive}\n disabled={disabled}\n >\n <Box display=\"flex\" alignItems=\"center\">\n {iconName && (\n <Icon name={iconName} size=\"mini\" pr={300} {...iconProps} />\n )}\n {value}\n </Box>\n </Token>\n );\n }\n\n renderTokens(tokens: TypeTokenSpec[]): React.ReactNode {\n return tokens.map<React.ReactNode>((token) => (\n <div key={token.id} className=\"TokenInput-token\">\n {this.renderToken(token)}\n </div>\n ));\n }\n\n override render() {\n const {\n autoFocus,\n autocomplete,\n disabled,\n isInvalid,\n hasWarning,\n id,\n name,\n placeholder,\n required,\n elemBefore,\n elemAfter,\n maxLength,\n ariaDescribedby,\n ariaLabel,\n innerRef,\n // These functions are used in the class functions above, but need to be extracted in order for `rest` to be correct\n /* eslint-disable @typescript-eslint/no-unused-vars */\n value,\n onAddToken,\n onRemoveToken,\n onChangeTokens,\n onClickToken,\n onBlur,\n onChange,\n onFocus,\n onKeyDown,\n onKeyUp,\n onPaste,\n /* eslint-enable @typescript-eslint/no-unused-vars */\n inputProps = {},\n qa = {},\n tokens,\n ...rest\n } = this.props;\n const { state } = this;\n return (\n <Container\n hasBeforeElement={!!elemBefore}\n hasAfterElement={!!elemAfter}\n disabled={disabled}\n invalid={!!isInvalid}\n warning={hasWarning}\n focused={state.hasFocus}\n {...rest}\n >\n {elemBefore && <Accessory before>{elemBefore}</Accessory>}\n\n {tokens && this.renderTokens(tokens)}\n\n <TokenScreenReaderStatus tokens={tokens} />\n\n <input\n aria-describedby={ariaDescribedby}\n aria-invalid={!!isInvalid}\n aria-label={ariaLabel}\n autoFocus={autoFocus}\n autoComplete={autocomplete}\n disabled={disabled}\n id={id}\n name={name}\n placeholder={placeholder}\n type=\"text\"\n required={required}\n value={state.value}\n maxLength={maxLength}\n onBlur={this.handleBlur}\n onChange={this.handleChangeText}\n onFocus={this.handleFocus}\n onKeyDown={this.handleKeyDown}\n onKeyUp={this.handleKeyUp}\n onPaste={this.handlePaste}\n ref={innerRef}\n data-qa-input={name || \"\"}\n data-qa-input-isdisabled={!!disabled}\n data-qa-input-isrequired={!!required}\n {...qa}\n {...inputProps}\n />\n\n {elemAfter && <Accessory after>{elemAfter}</Accessory>}\n </Container>\n );\n }\n}\n","import styled, { css } from \"styled-components\";\nimport { COMMON } from \"@sproutsocial/seeds-react-system-props\";\nimport { focusRing } from \"@sproutsocial/seeds-react-mixins\";\nimport type { TypeInputContainerProps } from \"@sproutsocial/seeds-react-input\";\n\ninterface TypeTokenInputContainerProps\n extends Pick<\n TypeInputContainerProps,\n \"hasBeforeElement\" | \"hasAfterElement\" | \"disabled\" | \"invalid\" | \"warning\"\n > {\n focused?: boolean;\n}\n\nconst Container = styled.div<TypeTokenInputContainerProps>`\n box-sizing: border-box;\n position: relative;\n display: flex;\n flex-wrap: wrap;\n align-items: center;\n align-content: center;\n cursor: text;\n width: 100%;\n border: 1px solid ${(props) => props.theme.colors.form.border.base};\n border-radius: ${(props) => props.theme.radii[500]};\n margin: 0;\n padding: ${(props) => props.theme.space[300]};\n padding-top: ${(props) => props.theme.space[200]};\n background-color: ${(props) => props.theme.colors.form.background.base};\n color: ${(props) => props.theme.colors.text.body};\n transition: border-color ${(props) => props.theme.duration.fast}\n ${(props) => props.theme.easing.ease_in},\n box-shadow ${(props) => props.theme.duration.fast}\n ${(props) => props.theme.easing.ease_in};\n ${(props) => props.theme.typography[200]};\n font-family: ${(props) => props.theme.fontFamily};\n font-weight: ${(props) => props.theme.fontWeights.normal};\n appearance: none;\n\n button {\n margin: ${(props) => props.theme.space[200]}\n ${(props) => props.theme.space[200]} 0 0;\n }\n\n input {\n ${(props) => props.theme.typography[200]};\n outline: none;\n border: none;\n flex: 1;\n padding: 0;\n padding-top: ${(props) => props.theme.space[100]};\n margin: ${(props) => props.theme.space[200]}\n ${(props) => props.theme.space[300]} ${(props) => props.theme.space[100]}\n 0;\n color: ${(props) => props.theme.colors.text.body};\n background-color: ${(props) => props.theme.colors.form.background.base};\n /** This matches the height of the token so size does not change as tokens are added */\n min-height: 20px;\n\n &::-webkit-search-cancel-button {\n appearance: none;\n }\n\n /* Explicitly removes double focus ring in environments where box-shadow focus styles have been specified (Seeds). Focus is passed up from the input to the parent container. */\n &:focus {\n box-shadow: none;\n }\n\n /* https://stackoverflow.com/questions/14007655/remove-ie10s-clear-field-x-button-on-certain-inputs */\n &::-ms-clear {\n display: none;\n }\n\n /* Fix for red ring when input is marked required in Firefox */\n &:not(output):not(:focus):-moz-ui-invalid {\n box-shadow: none;\n }\n\n &::placeholder {\n color: ${(props) => props.theme.colors.form.placeholder.base};\n font-style: italic;\n }\n\n ${(props) =>\n props.disabled &&\n css`\n opacity: 0.4;\n\n cursor: not-allowed;\n `}\n }\n\n ${(props) =>\n props.hasBeforeElement &&\n css`\n padding-left: 40px;\n `}\n\n ${(props) =>\n props.hasAfterElement &&\n css`\n padding-right: 40px;\n `}\n\n\n ${(props) =>\n props.disabled &&\n css`\n opacity: 0.4;\n\n cursor: not-allowed;\n `}\n\n ${(props) =>\n props.focused &&\n css`\n ${focusRing}\n `}\n\n ${(props) =>\n props.invalid &&\n css`\n border-color: ${(props) => props.theme.colors.form.border.error};\n `}\n\n\t${(props) =>\n props.warning &&\n css`\n border-color: ${(props) => props.theme.colors.form.border.warning};\n `}\n\n ${COMMON}\n`;\n\nContainer.displayName = \"TokenInputContainer\";\n\nexport default Container;\n","import uniqueId from \"lodash.uniqueid\";\nimport type { TypeTokenSpec } from \"./\";\n\nexport const asTokenSpec = (text: string): TypeTokenSpec => ({\n id: uniqueId(`${text}-`),\n value: text.trim(),\n});\n\nconst KeyNameToRegExpChar: { [key: string]: string } = {\n Enter: \"\\\\n\",\n};\n\nexport const delimitersAsRegExp = (delimiters: string[] | null | undefined) => {\n if (!delimiters) return /[,\\n]/;\n const chars = delimiters\n .map(\n (key) =>\n KeyNameToRegExpChar[key as keyof typeof KeyNameToRegExpChar] || key\n )\n .join(\"\");\n return RegExp(`[${chars}]`);\n};\n","import React from \"react\";\nimport { useEffect, useRef, useState } from \"react\";\nimport { VisuallyHidden } from \"@sproutsocial/seeds-react-visually-hidden\";\nimport type { TypeTokenInputProps } from \"./\";\n\nfunction usePrevious(value: TypeTokenInputProps[\"tokens\"]) {\n const ref = useRef<TypeTokenInputProps[\"tokens\"]>();\n\n useEffect(() => {\n ref.current = value;\n });\n\n return ref.current;\n}\n\nexport const TokenScreenReaderStatus = ({\n tokens,\n}: {\n tokens: TypeTokenInputProps[\"tokens\"];\n}) => {\n const prevTokens = usePrevious(tokens);\n const [tokenStatus, setTokenStatus] = useState(\"\");\n\n // TODO: Use callbacks so consumers can pass localized messaging to the screen reader\n useEffect(() => {\n if (prevTokens && tokens) {\n if (prevTokens.length > tokens.length) {\n setTokenStatus(\n `${\n prevTokens.filter((item) => tokens.indexOf(item) === -1)[0]?.value\n } has been removed`\n );\n }\n\n if (prevTokens.length < tokens.length) {\n setTokenStatus(`${tokens[tokens.length - 1]?.value} has been added.`);\n }\n }\n }, [prevTokens, tokens, tokenStatus]);\n\n return (\n <VisuallyHidden as=\"div\" role=\"status\">\n {tokenStatus}\n </VisuallyHidden>\n );\n};\n","import styled from \"styled-components\";\nimport { visuallyHidden } from \"@sproutsocial/seeds-react-mixins\";\nimport type { ComponentPropsWithRef } from \"react\";\n\nconst StyledVisuallyHidden = styled.span`\n ${visuallyHidden}\n`;\n\nexport type VisuallyHiddenProps = ComponentPropsWithRef<\n typeof StyledVisuallyHidden\n>;\n\n// This wrapper component is needed for react-docgen to work.\n// It has issues with generating docs for the styled component directly.\nexport const VisuallyHidden = (props: VisuallyHiddenProps) => {\n return <StyledVisuallyHidden {...props} />;\n};\n","import * as React from \"react\";\nimport type {\n TypeStyledComponentsCommonProps,\n TypeSystemCommonProps,\n} from \"@sproutsocial/seeds-react-system-props\";\nimport type {\n TypeIconProps,\n TypeIconName,\n} from \"@sproutsocial/seeds-react-icon\";\n\nexport interface TypeTokenSpec {\n id: string;\n iconName?: TypeIconName;\n iconProps?: Omit<TypeIconProps, \"name\">;\n value: string;\n valid?: boolean;\n}\n\nexport interface TypeBaseTokenInputProps {\n /** ID of the form element, should match the \"for\" value of the associated label */\n id: string;\n name: string;\n iconName?: TypeIconName;\n\n /** Array of delimiter key names */\n delimiters?: string[];\n\n /** Current input text. Required when controlling the input text */\n value?: string;\n\n /** Current focus state. Required when controlling the focus via onFocus and onBlur */\n hasFocus?: boolean;\n\n /** Id of the currently selected token */\n activeToken?: string;\n\n /** Array of current tokens */\n tokens?: TypeTokenSpec[];\n\n /** Standard control of changing tokens. For fine-grain control use onAddToken and onRemoveToken, instead */\n onChangeTokens?: (tokens: TypeTokenSpec[]) => void;\n\n /** Fine-grained control of adding tokens. Use with onRemoveToken instead of onChangeTokens */\n onAddToken?: (tokenSpec: TypeTokenSpec) => void;\n\n /** Fine-grained control of removing tokens. Use with onAddToken instead of onChangeTokens */\n onRemoveToken?: (tokenId: string) => void;\n\n /** Controls clicking on a token. When absent, clicking a token removes itself */\n onClickToken?: (\n e: React.SyntheticEvent<HTMLButtonElement>,\n tokenId: string\n ) => void;\n\n /** Fine-grained control of the input text used to create tokens */\n onChange?: (e: React.SyntheticEvent<HTMLInputElement>, value: string) => void;\n\n /** Fine-grained control of pasted text */\n onPaste?: (e: React.ClipboardEvent<HTMLInputElement>, value: string) => void;\n onBlur?: (e: React.FocusEvent<HTMLInputElement>) => void;\n onFocus?: (e: React.FocusEvent<HTMLInputElement>) => void;\n onKeyDown?: (e: React.KeyboardEvent<HTMLInputElement>, value: string) => void;\n onKeyUp?: (e: React.KeyboardEvent<HTMLInputElement>, value: string) => void;\n\n /** Attribute used to associate other elements that describe the Input, like an error */\n ariaDescribedby?: string;\n\n /** Label used to describe the input if not used with an accompanying visual label */\n ariaLabel?: string;\n\n /** Placeholder text for when value is undefined or empty */\n placeholder?: string;\n\n /** Will autofocus the element when mounted to the DOM */\n autoFocus?: boolean;\n\n /** HTML disabled attribute */\n disabled?: boolean;\n\n /** Whether or not the current contents of the input are invalid */\n isInvalid?: boolean;\n\n /** Whether or not the current contents of the input has any warnings */\n hasWarning?: boolean;\n\n /** HTML required attribute */\n required?: boolean;\n\n /** 16x16 element, such as an icon */\n elemBefore?: React.ReactNode;\n\n /** 16x16 element, such as an icon */\n elemAfter?: React.ReactNode;\n\n /** Max input text length */\n maxLength?: number;\n\n /** Max length of the token */\n tokenMaxLength?: number;\n\n /** Props to spread onto the underlying input element */\n inputProps?: React.ComponentPropsWithoutRef<\"input\">;\n\n /** Used to get a reference to the underlying element */\n innerRef?: React.Ref<HTMLInputElement>;\n qa?: object;\n\n /** Browser autocomplete support */\n autocomplete?: string;\n}\n\nexport interface TypeTokenInputProps\n extends TypeBaseTokenInputProps,\n TypeStyledComponentsCommonProps,\n TypeSystemCommonProps,\n Omit<\n React.ComponentPropsWithoutRef<\"div\">,\n keyof TypeBaseTokenInputProps | \"color\"\n > {}\n"],"mappings":"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","names":["React","styled","props","uniqueId","import_react","import_styled_components","import_seeds_react_mixins","styled","import_jsx_runtime","import_jsx_runtime","Token","Box","Icon","React"]}
package/jest.config.js ADDED
@@ -0,0 +1,9 @@
1
+ const baseConfig = require("@sproutsocial/seeds-testing");
2
+
3
+ /** * @type {import('jest').Config} */
4
+ const config = {
5
+ ...baseConfig,
6
+ displayName: "seeds-react-token-input",
7
+ };
8
+
9
+ module.exports = config;