UNPKG

@acrodata/gradient-picker

Version:

A powerful and beautiful gradient picker.

79 lines 11.3 kB
import { resolveColorInterp, resolvePosition, resolveStops, split } from './utils'; const set = new Set(['from', 'in', 'at']); function resolvePrefix(k, props, start, end) { switch (k) { case 'from': return { angle: props.slice(start, end).join(' ') }; case 'at': return { position: resolvePosition(props.slice(start, end).join(' ')) }; case 'in': { const arr = props.slice(start, end); return { color: resolveColorInterp(arr.join(' ')), }; } default: return null; } } export function parseConicGradient(input) { if (!/(repeating-)?conic-gradient/.test(input)) throw new SyntaxError(`could not find syntax for this item: ${input}`); const [, repeating, props] = input .replace(/[\n\t]/g, '') .match(/(repeating-)?conic-gradient\((.+)\)/); const result = { repeating: Boolean(repeating), angle: '0deg', position: { x: { type: 'keyword', value: 'center' }, y: { type: 'keyword', value: 'center' }, }, stops: [], }; const properties = split(props).map(v => v.trim()); const prefix = split(properties[0], /\s+/); let k = ''; let j = 0; for (let i = 0, n = prefix.length; i < n; i++) { if (set.has(prefix[i])) { if (i > 0) { Object.assign(result, resolvePrefix(k, prefix, j, i)); } k = prefix[i]; j = i + 1; } } if (k) { Object.assign(result, resolvePrefix(k, prefix, j, prefix.length)); properties.shift(); } return { ...result, stops: resolveStops(properties) }; } export function stringifyConicGradient(input) { const { repeating, angle, position, color, stops } = input; const type = repeating ? 'repeating-conic-gradient' : 'conic-gradient'; const prefixArr = []; if (angle.trim()) { prefixArr.push(`from ${angle}`); } const posX = position.x.value; const posY = position.y.value; const pos = posX.trim() || posY.trim() ? 'at ' + `${posX} ${posY}`.trim() : ''; if (pos) { prefixArr.push(pos); } if (color && color.space) { prefixArr.push(`in ${color.space} ${color.method || ''}`.trim()); } const props = []; if (prefixArr.length > 0) { props.push(prefixArr.join(' ')); } const stopsStr = stops .map(s => `${s.color} ${s.offset?.value}${s.offset?.unit}`.trim()) .join(', '); props.push(stopsStr); return `${type}(${props.join(', ')})`; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"conic.js","sourceRoot":"","sources":["../../../../../projects/gradient-picker/src/lib/parser/conic.ts"],"names":[],"mappings":"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","sourcesContent":["import { Color, ColorStop, PositionPropertyValue } from './type';\nimport { resolveColorInterp, resolvePosition, resolveStops, split } from './utils';\n\nexport interface ConicGradientResult {\n  repeating: boolean;\n  angle: string;\n  position: {\n    x: PositionPropertyValue;\n    y: PositionPropertyValue;\n  };\n  color?: Color;\n  stops: ColorStop[];\n}\n\nconst set = new Set(['from', 'in', 'at']);\n\nfunction resolvePrefix(k: string, props: string[], start: number, end: number) {\n  switch (k) {\n    case 'from':\n      return { angle: props.slice(start, end).join(' ') };\n    case 'at':\n      return { position: resolvePosition(props.slice(start, end).join(' ')) };\n    case 'in': {\n      const arr = props.slice(start, end);\n      return {\n        color: resolveColorInterp(arr.join(' ')),\n      };\n    }\n    default:\n      return null;\n  }\n}\n\nexport function parseConicGradient(input: string): ConicGradientResult {\n  if (!/(repeating-)?conic-gradient/.test(input))\n    throw new SyntaxError(`could not find syntax for this item: ${input}`);\n\n  const [, repeating, props] = input\n    .replace(/[\\n\\t]/g, '')\n    .match(/(repeating-)?conic-gradient\\((.+)\\)/)!;\n  const result: ConicGradientResult = {\n    repeating: Boolean(repeating),\n    angle: '0deg',\n    position: {\n      x: { type: 'keyword', value: 'center' },\n      y: { type: 'keyword', value: 'center' },\n    },\n    stops: [],\n  };\n\n  const properties = split(props).map(v => v.trim());\n\n  const prefix = split(properties[0], /\\s+/);\n\n  let k = '';\n  let j = 0;\n  for (let i = 0, n = prefix.length; i < n; i++) {\n    if (set.has(prefix[i])) {\n      if (i > 0) {\n        Object.assign(result, resolvePrefix(k, prefix, j, i));\n      }\n      k = prefix[i];\n      j = i + 1;\n    }\n  }\n\n  if (k) {\n    Object.assign(result, resolvePrefix(k, prefix, j, prefix.length));\n    properties.shift();\n  }\n\n  return { ...result, stops: resolveStops(properties) };\n}\n\nexport function stringifyConicGradient(input: ConicGradientResult) {\n  const { repeating, angle, position, color, stops } = input;\n\n  const type = repeating ? 'repeating-conic-gradient' : 'conic-gradient';\n\n  const prefixArr: string[] = [];\n\n  if (angle.trim()) {\n    prefixArr.push(`from ${angle}`);\n  }\n\n  const posX = position.x.value;\n  const posY = position.y.value;\n  const pos = posX.trim() || posY.trim() ? 'at ' + `${posX} ${posY}`.trim() : '';\n  if (pos) {\n    prefixArr.push(pos);\n  }\n\n  if (color && color.space) {\n    prefixArr.push(`in ${color.space} ${color.method || ''}`.trim());\n  }\n\n  const props: string[] = [];\n\n  if (prefixArr.length > 0) {\n    props.push(prefixArr.join(' '));\n  }\n\n  const stopsStr = stops\n    .map(s => `${s.color} ${s.offset?.value}${s.offset?.unit}`.trim())\n    .join(', ');\n\n  props.push(stopsStr);\n\n  return `${type}(${props.join(', ')})`;\n}\n"]}