UNPKG

@dekk-ui/color-input-field

Version:
206 lines (178 loc) 11.3 kB
/*! * Copyright (c) Dekk * @author Gregor Adams * @license MIT * @name @dekk-ui/color-input-field * @version 0.8.0 * @see git+https://github.com/dekk-app/design-system.git */ import { StyledCombinedInput } from '@dekk-ui/combined-input-field/styled'; import { StyledInput as StyledInput$1 } from '@dekk-ui/input-field/styled'; import React, { forwardRef, useState, useCallback } from 'react'; import _styled from '@emotion/styled/base'; import { focus } from '@dekk-ui/focus-ring'; import { StyledInput } from '@dekk-ui/input-field'; import { pxToRem } from '@dekk-ui/utils/px-to-rem'; import { css } from '@emotion/react'; function getDefaultExportFromCjs (x) { return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x; } var _extends$1 = {exports: {}}; (function (module) { function _extends() { module.exports = _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; module.exports["default"] = module.exports, module.exports.__esModule = true; return _extends.apply(this, arguments); } module.exports = _extends; module.exports["default"] = module.exports, module.exports.__esModule = true; }(_extends$1)); var _extends = /*@__PURE__*/getDefaultExportFromCjs(_extends$1.exports); /*! ***************************************************************************** Copyright (c) Microsoft Corporation. Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted. THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. ***************************************************************************** */ function __rest(s, e) { var t = {}; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0) t[p] = s[p]; if (s != null && typeof Object.getOwnPropertySymbols === "function") for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) { if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i])) t[p[i]] = s[p[i]]; } return t; } const StyledColorInput = /*#__PURE__*/_styled(StyledInput, process.env.NODE_ENV === "production" ? { target: "elom7ma0" } : { target: "elom7ma0", label: "StyledColorInput" })("&[type=\"color\"]{padding:0;", ({ theme }) => /*#__PURE__*/css("width:", pxToRem(theme.space.l), ";border-radius:0 ", pxToRem(theme.radius.s), " ", pxToRem(theme.radius.s), " 0;&::-webkit-color-swatch{border-radius:0 ", pxToRem(theme.radius.s), " ", pxToRem(theme.radius.s), " 0;border:1px solid ", theme.ui.outline["1"], ";}&::-moz-color-swatch{border-radius:0 ", pxToRem(theme.radius.s), " ", pxToRem(theme.radius.s), " 0;border:1px solid ", theme.ui.outline["1"], ";}&:focus-visible{&::-webkit-color-swatch{", focus, ";}&::-moz-color-swatch{", focus, ";}}" + (process.env.NODE_ENV === "production" ? "" : ";label:StyledColorInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInN0eWxlZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUF5QnVCIiwiZmlsZSI6InN0eWxlZC50cyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGZvY3VzIH0gZnJvbSBcIkBkZWtrLXVpL2ZvY3VzLXJpbmdcIjtcbmltcG9ydCB7IFN0eWxlZElucHV0IH0gZnJvbSBcIkBkZWtrLXVpL2lucHV0LWZpZWxkXCI7XG5pbXBvcnQgeyBweFRvUmVtIH0gZnJvbSBcIkBkZWtrLXVpL3V0aWxzL3B4LXRvLXJlbVwiO1xuaW1wb3J0IHsgY3NzIH0gZnJvbSBcIkBlbW90aW9uL3JlYWN0XCI7XG5pbXBvcnQgc3R5bGVkIGZyb20gXCJAZW1vdGlvbi9zdHlsZWRcIjtcbmV4cG9ydCBjb25zdCBTdHlsZWRDb2xvcklucHV0ID0gc3R5bGVkKFN0eWxlZElucHV0KSBgXG5cdCZbdHlwZT1cImNvbG9yXCJdIHtcblx0XHRwYWRkaW5nOiAwO1xuXG5cdFx0Jjo6LXdlYmtpdC1jb2xvci1zd2F0Y2gge1xuXHRcdFx0b3ZlcmZsb3c6IGhpZGRlbjtcblx0XHR9XG5cblx0XHQmOjotbW96LWNvbG9yLXN3YXRjaCB7XG5cdFx0XHRvdmVyZmxvdzogaGlkZGVuO1xuXHRcdH1cblxuXHRcdCY6Oi13ZWJraXQtY29sb3Itc3dhdGNoLXdyYXBwZXIge1xuXHRcdFx0cGFkZGluZzogMDtcblx0XHR9XG5cblx0XHQmOmZvY3VzIHtcblx0XHRcdG91dGxpbmU6IDA7XG5cdFx0fVxuXG5cdFx0JHsoeyB0aGVtZSB9KSA9PiBjc3MgYFxuXHRcdFx0d2lkdGg6ICR7cHhUb1JlbSh0aGVtZS5zcGFjZS5sKX07XG5cdFx0XHRib3JkZXItcmFkaXVzOiAwICR7cHhUb1JlbSh0aGVtZS5yYWRpdXMucyl9ICR7cHhUb1JlbSh0aGVtZS5yYWRpdXMucyl9IDA7XG5cblx0XHRcdCY6Oi13ZWJraXQtY29sb3Itc3dhdGNoIHtcblx0XHRcdFx0Ym9yZGVyLXJhZGl1czogMCAke3B4VG9SZW0odGhlbWUucmFkaXVzLnMpfSAke3B4VG9SZW0odGhlbWUucmFkaXVzLnMpfSAwO1xuXHRcdFx0XHRib3JkZXI6IDFweCBzb2xpZCAke3RoZW1lLnVpLm91dGxpbmVbXCIxXCJdfTtcblx0XHRcdH1cblxuXHRcdFx0Jjo6LW1vei1jb2xvci1zd2F0Y2gge1xuXHRcdFx0XHRib3JkZXItcmFkaXVzOiAwICR7cHhUb1JlbSh0aGVtZS5yYWRpdXMucyl9ICR7cHhUb1JlbSh0aGVtZS5yYWRpdXMucyl9IDA7XG5cdFx0XHRcdGJvcmRlcjogMXB4IHNvbGlkICR7dGhlbWUudWkub3V0bGluZVtcIjFcIl19O1xuXHRcdFx0fVxuXG5cdFx0XHQmOmZvY3VzLXZpc2libGUge1xuXHRcdFx0XHQmOjotd2Via2l0LWNvbG9yLXN3YXRjaCB7XG5cdFx0XHRcdFx0JHtmb2N1c307XG5cdFx0XHRcdH1cblxuXHRcdFx0XHQmOjotbW96LWNvbG9yLXN3YXRjaCB7XG5cdFx0XHRcdFx0JHtmb2N1c307XG5cdFx0XHRcdH1cblx0XHRcdH1cblx0XHRgfTtcblx0fVxuYDtcbi8vIyBzb3VyY2VNYXBwaW5nVVJMPXN0eWxlZC5qcy5tYXAiXX0= */"), ";&::-webkit-color-swatch{overflow:hidden;}&::-moz-color-swatch{overflow:hidden;}&::-webkit-color-swatch-wrapper{padding:0;}&:focus{outline:0;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInN0eWxlZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLb0QiLCJmaWxlIjoic3R5bGVkLnRzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZm9jdXMgfSBmcm9tIFwiQGRla2stdWkvZm9jdXMtcmluZ1wiO1xuaW1wb3J0IHsgU3R5bGVkSW5wdXQgfSBmcm9tIFwiQGRla2stdWkvaW5wdXQtZmllbGRcIjtcbmltcG9ydCB7IHB4VG9SZW0gfSBmcm9tIFwiQGRla2stdWkvdXRpbHMvcHgtdG8tcmVtXCI7XG5pbXBvcnQgeyBjc3MgfSBmcm9tIFwiQGVtb3Rpb24vcmVhY3RcIjtcbmltcG9ydCBzdHlsZWQgZnJvbSBcIkBlbW90aW9uL3N0eWxlZFwiO1xuZXhwb3J0IGNvbnN0IFN0eWxlZENvbG9ySW5wdXQgPSBzdHlsZWQoU3R5bGVkSW5wdXQpIGBcblx0Jlt0eXBlPVwiY29sb3JcIl0ge1xuXHRcdHBhZGRpbmc6IDA7XG5cblx0XHQmOjotd2Via2l0LWNvbG9yLXN3YXRjaCB7XG5cdFx0XHRvdmVyZmxvdzogaGlkZGVuO1xuXHRcdH1cblxuXHRcdCY6Oi1tb3otY29sb3Itc3dhdGNoIHtcblx0XHRcdG92ZXJmbG93OiBoaWRkZW47XG5cdFx0fVxuXG5cdFx0Jjo6LXdlYmtpdC1jb2xvci1zd2F0Y2gtd3JhcHBlciB7XG5cdFx0XHRwYWRkaW5nOiAwO1xuXHRcdH1cblxuXHRcdCY6Zm9jdXMge1xuXHRcdFx0b3V0bGluZTogMDtcblx0XHR9XG5cblx0XHQkeyh7IHRoZW1lIH0pID0+IGNzcyBgXG5cdFx0XHR3aWR0aDogJHtweFRvUmVtKHRoZW1lLnNwYWNlLmwpfTtcblx0XHRcdGJvcmRlci1yYWRpdXM6IDAgJHtweFRvUmVtKHRoZW1lLnJhZGl1cy5zKX0gJHtweFRvUmVtKHRoZW1lLnJhZGl1cy5zKX0gMDtcblxuXHRcdFx0Jjo6LXdlYmtpdC1jb2xvci1zd2F0Y2gge1xuXHRcdFx0XHRib3JkZXItcmFkaXVzOiAwICR7cHhUb1JlbSh0aGVtZS5yYWRpdXMucyl9ICR7cHhUb1JlbSh0aGVtZS5yYWRpdXMucyl9IDA7XG5cdFx0XHRcdGJvcmRlcjogMXB4IHNvbGlkICR7dGhlbWUudWkub3V0bGluZVtcIjFcIl19O1xuXHRcdFx0fVxuXG5cdFx0XHQmOjotbW96LWNvbG9yLXN3YXRjaCB7XG5cdFx0XHRcdGJvcmRlci1yYWRpdXM6IDAgJHtweFRvUmVtKHRoZW1lLnJhZGl1cy5zKX0gJHtweFRvUmVtKHRoZW1lLnJhZGl1cy5zKX0gMDtcblx0XHRcdFx0Ym9yZGVyOiAxcHggc29saWQgJHt0aGVtZS51aS5vdXRsaW5lW1wiMVwiXX07XG5cdFx0XHR9XG5cblx0XHRcdCY6Zm9jdXMtdmlzaWJsZSB7XG5cdFx0XHRcdCY6Oi13ZWJraXQtY29sb3Itc3dhdGNoIHtcblx0XHRcdFx0XHQke2ZvY3VzfTtcblx0XHRcdFx0fVxuXG5cdFx0XHRcdCY6Oi1tb3otY29sb3Itc3dhdGNoIHtcblx0XHRcdFx0XHQke2ZvY3VzfTtcblx0XHRcdFx0fVxuXHRcdFx0fVxuXHRcdGB9O1xuXHR9XG5gO1xuLy8jIHNvdXJjZU1hcHBpbmdVUkw9c3R5bGVkLmpzLm1hcCJdfQ== */")); const hexKeys = new Set(["Digit1", "Digit2", "Digit3", "Digit4", "Digit5", "Digit6", "Digit7", "Digit8", "Digit9", "Digit0", "KeyA", "KeyB", "KeyC", "KeyD", "KeyE", "KeyF"]); const modKeys = new Set(["Backspace", "Tab", "ArrowLeft", "ArrowUp", "ArrowRight", "ArrowDown"]); const copyAndPasteKeys = new Set(["KeyC", "KeyV", "KeyX"]); const getSelectionText = () => { const activeElement = document.activeElement; if (activeElement.tagName === "INPUT") { const { type, value, selectionStart, selectionEnd } = activeElement; const match = /^(?:text|search|password|tel|url)$/i.exec(type); const hasStart = typeof selectionStart === "number"; if (match && hasStart) { return value.slice(selectionStart, selectionEnd); } } else if (window.getSelection) { return window.getSelection().toString(); } return ""; }; const ColorInput = /*#__PURE__*/forwardRef((_a, ref) => { var { defaultValue, disabled, width, fullWidth, onChange, onBlur, onKeyDown } = _a, props = __rest(_a, ["defaultValue", "disabled", "width", "fullWidth", "onChange", "onBlur", "onKeyDown"]); const [value, setValue] = useState(defaultValue); const [valid, setValid] = useState(null); const handleChange = useCallback(event_ => { const { target } = event_; setValid(target.validity.valid); if (target.value.length > 7 && target.value.startsWith("#")) { setValid(true); const newValue = target.value.slice(0, 7); setValue(newValue); } else if (target.value.length > 1 && target.value.startsWith("#")) { setValue(target.value); } else if (target.value === "#") { setValid(null); setValue(""); } else if (target.value.length === 0) { setValue(target.value); } else { setValue(`#${target.value.slice(0, 6)}`); } if (onChange) { onChange(event_); } }, [onChange]); const handleKeyDown = useCallback(event_ => { const selectionText = getSelectionText(); if (!hexKeys.has(event_.code) && !modKeys.has(event_.code) && !(event_.metaKey && copyAndPasteKeys.has(event_.code))) { event_.preventDefault(); } else if (!modKeys.has(event_.code) && (event_.currentTarget.value.length > 6 && selectionText.length === 0 || selectionText.includes("#"))) { event_.preventDefault(); } if (onKeyDown) { onKeyDown(event_); } }, [onKeyDown]); const handleBlur = useCallback(event_ => { const { target } = event_; if (target.value.length === 4 && target.value.startsWith("#")) { const [, _1, _2, _3] = target.value.split(""); setValid(true); setValue(`#${_1}${_1}${_2}${_2}${_3}${_3}`); } if (onBlur) { onBlur(event_); } }, [onBlur]); return /*#__PURE__*/React.createElement(StyledCombinedInput, { isInvalid: valid === null ? undefined : !valid, fullWidth: fullWidth, width: width, isDisabled: disabled }, /*#__PURE__*/React.createElement(StyledInput$1, _extends({}, props, { autoComplete: "off", autoCorrect: "off", autoCapitalize: "off", spellCheck: "false", pattern: "^#[a-f0-9]{6}", value: value, disabled: disabled, invalid: valid === null ? undefined : !valid, onChange: handleChange, onKeyDown: handleKeyDown, onBlur: handleBlur })), /*#__PURE__*/React.createElement(StyledColorInput, { ref: ref, value: value, type: "color", onChange: handleChange })); }); export { ColorInput };