ngx-input-color
Version:
Angular color input component and color picker (with HSL, HSV, RGB, CMYK, HEX, alpha, eye-dropper, etc)
110 lines • 14.3 kB
JavaScript
export function buildGradientFromStops(stops, type = 'linear', rotation = 0) {
if (!stops || stops.length === 0)
return '';
const sorted = [...stops].sort((a, b) => a.value - b.value);
const parts = [];
for (const stop of sorted) {
const value = Math.max(0, Math.min(stop.value, 100));
parts.push(`${stop.color} ${value}%`);
}
let f = '';
if (type === 'linear') {
f = `${rotation}deg, `;
}
else {
f = 'circle, ';
}
return `${type}-gradient(${f}${parts.join(', ')})`;
}
export function generateRandomColor() {
const letters = '0123456789ABCDEF';
let color = '#';
for (let i = 0; i < 6; i++) {
color += letters[Math.floor(Math.random() * 16)];
}
return color;
}
export function isValidGradient(value) {
// Accepts linear-gradient or radial-gradient with any color format
return /^(\s*)(linear|radial)-gradient\s*\(/i.test(value);
}
export function parseGradient(value) {
let type = 'linear';
let rotation = 90;
let stops = [];
let valid = false;
let match = value.match(/^(\s*)(linear|radial)-gradient\s*\((.*)\)$/i);
if (!match)
return { type, rotation, stops, valid };
type = match[2];
let content = match[3];
// Split by commas, but ignore commas inside parentheses (for rgb, hsl, etc)
let parts = [];
let buf = '', depth = 0;
for (let c of content) {
if (c === '(')
depth++;
if (c === ')')
depth--;
if (c === ',' && depth === 0) {
parts.push(buf.trim());
buf = '';
}
else {
buf += c;
}
}
if (buf)
parts.push(buf.trim());
// First part may be angle/direction (for linear) or shape/position (for radial)
let first = parts[0];
let colorStopStart = 0;
if (type === 'linear') {
let angleMatch = first.match(/^(\d+)(deg)?$/i);
if (angleMatch) {
rotation = parseInt(angleMatch[1], 10);
colorStopStart = 1;
}
else if (/to /.test(first)) {
// e.g. 'to right', 'to bottom left' (optional: map to degree)
// You can add mapping if needed
colorStopStart = 1;
}
}
else if (type === 'radial') {
// e.g. 'circle at center', 'ellipse at top left', etc
if (!/^(#|rgb|hsl|[a-z])/i.test(first))
colorStopStart = 1;
}
// Color stop regex: supports hex, rgb(a), hsl(a), color names, with optional position
const colorStopRegex = /((#([0-9a-fA-F]{3,8}))|(rgba?\([^\)]+\))|(hsla?\([^\)]+\))|([a-zA-Z]+))(\s+([\d.]+%?|[\d.]+px|[\d.]+em))?/;
for (let i = colorStopStart; i < parts.length; i++) {
let stopPart = parts[i];
let m = stopPart.match(colorStopRegex);
if (m) {
let color = m[1];
let posStr = m[8];
let value = 0;
if (posStr) {
if (posStr.endsWith('%'))
value = parseFloat(posStr);
else
value = parseFloat(posStr); // px/em: you may want to normalize or keep as is
}
else {
value = i === colorStopStart ? 0 : 100;
}
stops.push({ color, value, id: generateId(stops) });
}
}
valid = stops.length >= 2;
return { type, rotation, stops, valid };
}
function generateId(rangeValues) {
let id = 'ngx-thumb-' + Math.random().toString(36).substring(2, 9);
if (rangeValues.findIndex((x) => x.id == id) >= 0) {
return generateId(rangeValues);
}
return id;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"build-gradient.js","sourceRoot":"","sources":["../../../../projects/ngx-input-color/src/utils/build-gradient.ts"],"names":[],"mappings":"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","sourcesContent":["import { GradientStop, GradientType } from '../models/GradientStop';\r\n\r\nexport function buildGradientFromStops(stops: GradientStop[], type: GradientType = 'linear', rotation = 0): string {\r\n  if (!stops || stops.length === 0) return '';\r\n\r\n  const sorted = [...stops].sort((a, b) => a.value - b.value);\r\n\r\n  const parts: string[] = [];\r\n\r\n  for (const stop of sorted) {\r\n    const value = Math.max(0, Math.min(stop.value, 100));\r\n    parts.push(`${stop.color} ${value}%`);\r\n  }\r\n\r\n  let f = '';\r\n  if (type === 'linear') {\r\n    f = `${rotation}deg, `;\r\n  } else {\r\n    f = 'circle, ';\r\n  }\r\n\r\n  return `${type}-gradient(${f}${parts.join(', ')})`;\r\n}\r\n\r\nexport function generateRandomColor(): string {\r\n  const letters = '0123456789ABCDEF';\r\n  let color = '#';\r\n  for (let i = 0; i < 6; i++) {\r\n    color += letters[Math.floor(Math.random() * 16)];\r\n  }\r\n  return color;\r\n}\r\n\r\nexport function isValidGradient(value: string): boolean {\r\n  // Accepts linear-gradient or radial-gradient with any color format\r\n  return /^(\\s*)(linear|radial)-gradient\\s*\\(/i.test(value);\r\n}\r\n\r\nexport function parseGradient(value: string): {\r\n  type: GradientType;\r\n  rotation: number;\r\n  stops: GradientStop[];\r\n  valid: boolean;\r\n} {\r\n  let type: GradientType = 'linear';\r\n  let rotation = 90;\r\n  let stops: GradientStop[] = [];\r\n  let valid = false;\r\n  let match = value.match(/^(\\s*)(linear|radial)-gradient\\s*\\((.*)\\)$/i);\r\n  if (!match) return { type, rotation, stops, valid };\r\n  type = match[2] as GradientType;\r\n  let content = match[3];\r\n  // Split by commas, but ignore commas inside parentheses (for rgb, hsl, etc)\r\n  let parts = [];\r\n  let buf = '',\r\n    depth = 0;\r\n  for (let c of content) {\r\n    if (c === '(') depth++;\r\n    if (c === ')') depth--;\r\n    if (c === ',' && depth === 0) {\r\n      parts.push(buf.trim());\r\n      buf = '';\r\n    } else {\r\n      buf += c;\r\n    }\r\n  }\r\n  if (buf) parts.push(buf.trim());\r\n  // First part may be angle/direction (for linear) or shape/position (for radial)\r\n  let first = parts[0];\r\n  let colorStopStart = 0;\r\n  if (type === 'linear') {\r\n    let angleMatch = first.match(/^(\\d+)(deg)?$/i);\r\n    if (angleMatch) {\r\n      rotation = parseInt(angleMatch[1], 10);\r\n      colorStopStart = 1;\r\n    } else if (/to /.test(first)) {\r\n      // e.g. 'to right', 'to bottom left' (optional: map to degree)\r\n      // You can add mapping if needed\r\n      colorStopStart = 1;\r\n    }\r\n  } else if (type === 'radial') {\r\n    // e.g. 'circle at center', 'ellipse at top left', etc\r\n    if (!/^(#|rgb|hsl|[a-z])/i.test(first)) colorStopStart = 1;\r\n  }\r\n  // Color stop regex: supports hex, rgb(a), hsl(a), color names, with optional position\r\n  const colorStopRegex =\r\n    /((#([0-9a-fA-F]{3,8}))|(rgba?\\([^\\)]+\\))|(hsla?\\([^\\)]+\\))|([a-zA-Z]+))(\\s+([\\d.]+%?|[\\d.]+px|[\\d.]+em))?/;\r\n  for (let i = colorStopStart; i < parts.length; i++) {\r\n    let stopPart = parts[i];\r\n    let m = stopPart.match(colorStopRegex);\r\n    if (m) {\r\n      let color = m[1];\r\n      let posStr = m[8];\r\n      let value = 0;\r\n      if (posStr) {\r\n        if (posStr.endsWith('%')) value = parseFloat(posStr);\r\n        else value = parseFloat(posStr); // px/em: you may want to normalize or keep as is\r\n      } else {\r\n        value = i === colorStopStart ? 0 : 100;\r\n      }\r\n      stops.push({ color, value, id: generateId(stops) });\r\n    }\r\n  }\r\n  valid = stops.length >= 2;\r\n  return { type, rotation, stops, valid };\r\n}\r\nfunction generateId(rangeValues: GradientStop[]): string {\r\n  let id = 'ngx-thumb-' + Math.random().toString(36).substring(2, 9);\r\n  if (rangeValues.findIndex((x) => x.id == id) >= 0) {\r\n    return generateId(rangeValues);\r\n  }\r\n  return id;\r\n}\r\n"]}