From bc063a76e50addf0af5702ebb0e63afb03a6f294 Mon Sep 17 00:00:00 2001 From: Dean Tarisai Date: Mon, 22 Jul 2024 09:31:26 +0000 Subject: [PATCH] =?UTF-8?q?=F0=9F=90=9B=20fix(utilities):=20token()=20now?= =?UTF-8?q?=20converts=20arrays=20to=20objects=20beforehand=20to=20remove?= =?UTF-8?q?=20code=20ambiguity=20(feat=20#209)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit feat #209 --- src/utilities/index.js | 365 ++++++++++++++++++++++------------------- 1 file changed, 194 insertions(+), 171 deletions(-) diff --git a/src/utilities/index.js b/src/utilities/index.js index 1b807fba..0cabe6dd 100644 --- a/src/utilities/index.js +++ b/src/utilities/index.js @@ -39,6 +39,7 @@ import { isArray, neq, or, + lt, take, give, max, @@ -357,27 +358,23 @@ function token(color, options = undefined) { normalizeRgb = or(normalizeRgb, false); - /* - * * GLOBALS - * The following variables are valid if the token is a collection only - * - x is an array of channel keys from the source colorspace. If undefined it defaults to LCH - * - y is the array of channel values computed according to color tuple length - * - z is the alpha channel captured if it exists in the color token - * - * - * - */ - // if the color is an array turn it to an object - + /** + * an array of channel keys from the source colorspace. If undefined it defaults to LCH + * @type {string[]} + */ var x = gmchn(or(srcMode, targetMode)), - [y, z] = [ + /** + * @type {number[]} + * an array of channel values computed according to color tuple length + */ + y = or( or( or( // if the color is an array just take the values whilst optionally omitting the colorspace (if specified) and( and(isArray(color), eq(typeof color[0], "string")), - color.slice(1) + color?.slice(1) ), color ), @@ -386,21 +383,47 @@ function token(color, options = undefined) { x.map((c) => color[c]) ) ), - // check if the alpha channel is explicitly specified else cast 1 as the default - or( - or(and(eq(color?.length, 5), color[4]), 1), + undefined + ), + // step 2 get the alpha + + // check if the alpha channel is explicitly specified else cast 1 as the default + + /** + * the alpha channel captured if it exists in the color token + * @type{number} + */ + z = or( + or(and(eq(color?.length, 5), color[4]), 1), - // if its a string and has 8 or more characters (ignoring #) and is not a CSS named colortake the last two characters and convert them from hex + // if its a string and has 8 or more characters (ignoring #) and is not a CSS named colortake the last two characters and convert them from hex + and( and( - and( - and(eq(typeof color, "string"), not(colorsNamed[color])), - gte(color?.length, 8) - ), - parseInt(color?.slice(color?.length - 2), 16) - ) - ), - ]; + and(eq(typeof color, "string"), not(colorsNamed[color])), + gte(color?.length, 8) + ), + parseInt(color?.slice(color?.length - 2), 16) + ) + ); + + // convert the color to an object (including alpha) without the mode + color = or( + and( + y, + (() => { + let z = {}; + for (const [k, v] of entries(x)) { + z[v] = y[k]; + } + and(eq(y?.length, 4), (z["alpha"] = y[3])); + return z; + })() + ), + color + ); + console.log(color); + console.log(x, y, z); // convert the color to a target mode if it is specified /* @@ -415,151 +438,151 @@ function token(color, options = undefined) { * */ - function c2col() { - if (eq(kind, "obj")) { - return color; - } else if (eq(kind, "arr")) { - /** - * The result array if `kind` is `arr` - */ - var j = []; - for (const [k, v] of entries(x)) { - j[k] = color[v]; - } - - // if true, remove the - or( - and( - omitMode, - or( - and(eq(kind, "arr"), j.unshift(targetMode)), - and(eq(kind, "obj"), delete j?.mode) - ), - j - ) - ); - - // dont add alpha channel if true - or( - and( - omitAlpha, - or( - and(eq(kind, "arr"), j.push(z)), - and(eq(kind, "obj"), delete j?.alpha) - ) - ), - j - ); - - return j; - } - } - function c2num() { - const _ = cnv("rgb"); - - /** - * @type {number|string} - */ - // @ts-ignore - var s = ((255 * _["r"]) << 16) + ((255 * _["g"]) << 8) + 255 * _["b"]; - - return or( - and( - numType, - s.toString({ bin: 2, hex: 16, expo: 6, oct: 8 }[numType?.toLowerCase()]) - ), - s - ); - } - function c2hx() { - return { - boolean: or(and(eq(color, true), "#ffffff"), "#000000"), - number: num2c(), - object: formatHex(c2col()), - // @ts-ignore - string: or(colorsNamed?.(color?.toLowerCase()), formatHex(color)), - }[typeof color]; - } - function num2c() { - // Ported from chroma-js with slight modifications - // - // - return and( - and(eq(typeof color, "number"), gte(color, 0)), - lte(color, 0xffffff) - ) - ? { - // @ts-ignore - r: (color >> 16) / 255, - // @ts-ignore - g: ((color >> 8) & 0xff) / 255, - // @ts-ignore - b: (color & 0xff) / 255, - mode: "rgb", - } - : Error("unknown num color: " + color); - } - - /** - * * Conversion according to color token type - * - The conversion reassigns the color variable with the widely parseable format - * - If the color token is an array or plain object: - * * If it has a srcMode of rgb/lrgb we normalize the values if normalizeRgb is true - * * Finally we return the color as an object with values assigned using a forof loop - * - * - * - * - */ - or( - and(eq(typeof color, "number"), (color = num2c())), - or( - and(eq(typeof color, "string"), (color = cnv(targetMode))), - and( - eq(typeof color, "object"), - (() => { - /** - * This block only runs on objects not strings/numbers and boolean - */ - and( - eq(srcMode, and(or("rgb", "lrgb")), normalizeRgb), - (() => { - /** - * Normalize the color back to the rgb gamut supported by culori - * @type {boolean} - * */ - var s = y.some((c) => lt(1, Math.abs(c))); - y = or( - and( - s, - y.map((c) => c / 255) - ), - y - ); - })() - ); - - // reinitialize color to an empty object - color = {}; - - // then assign the alpha and colorspace properties - color = { alpha: z, mode: srcMode }; - - // assign channel keys with their values - for (const [k, v] of entries(x)) { - color[v] = y[k]; - } - })() - ) - ) - ); - - return { - obj: c2col, - arr: c2col, - str: c2hx, - num: c2num, - }[kind](); + // function c2col() { + // if (eq(kind, "obj")) { + // return color; + // } else if (eq(kind, "arr")) { + // /** + // * The result array if `kind` is `arr` + // */ + // var j = []; + // + + // // if true, remove the + // or( + // and( + // omitMode, + // or( + // and(eq(kind, "arr"), j.unshift(targetMode)), + // and(eq(kind, "obj"), delete j?.mode) + // ), + // j + // ) + // ); + + // // dont add alpha channel if true + // or( + // and( + // omitAlpha, + // or( + // and(eq(kind, "arr"), j.push(z)), + // and(eq(kind, "obj"), delete j?.alpha) + // ) + // ), + // j + // ); + + // return j; + // } + // } + // function c2num() { + // const _ = cnv("rgb"); + + // /** + // * @type {number|string} + // */ + // // @ts-ignore + // var s = ((255 * _["r"]) << 16) + ((255 * _["g"]) << 8) + 255 * _["b"]; + + // return or( + // and( + // numType, + // s.toString({ bin: 2, hex: 16, expo: 6, oct: 8 }[numType?.toLowerCase()]) + // ), + // s + // ); + // } + // function c2hx() { + // return { + // boolean: or(and(eq(color, true), "#ffffff"), "#000000"), + // number: num2c(), + // object: formatHex(c2col()), + // // @ts-ignore + // string: or(colorsNamed?.color, formatHex(color)), + // }[typeof color]; + // } + // function num2c() { + // // Ported from chroma-js with slight modifications + // // + // // + // return and( + // and(eq(typeof color, "number"), gte(color, 0)), + // lte(color, 0xffffff) + // ) + // ? { + // // @ts-ignore + // r: (color >> 16) / 255, + // // @ts-ignore + // g: ((color >> 8) & 0xff) / 255, + // // @ts-ignore + // b: (color & 0xff) / 255, + // mode: "rgb", + // } + // : Error("unknown num color: " + color); + // } + + // /** + // * * Conversion according to color token type + // * - The conversion reassigns the color variable with the widely parseable format + // * - If the color token is an array or plain object: + // * * If it has a srcMode of rgb/lrgb we normalize the values if normalizeRgb is true + // * * Finally we return the color as an object with values assigned using a forof loop + // * + // * + // * + // * + // */ + // or( + // and(eq(typeof color, "number"), (color = num2c())), + // or( + // and(eq(typeof color, "string"), (color = cnv(or(targetMode, srcMode)))), + // and( + // eq(typeof color, "object"), + // (() => { + // /** + // * This block only runs on objects not strings/numbers and boolean + // */ + // and( + // eq(srcMode, and(or("rgb", "lrgb")), normalizeRgb), + // (() => { + // /** + // * Normalize the color back to the rgb gamut supported by culori + // * @type {boolean} + // * */ + // var s = y.some((c) => lt(1, Math.abs(c))); + // y = or( + // and( + // s, + // y.map((c) => c / 255) + // ), + // y + // ); + // })() + // ); + + // // reinitialize color to an empty object + // color = {}; + + // // then assign the alpha and colorspace properties + // color = { alpha: z, mode: srcMode }; + + // // assign channel keys with their values + // for (const [k, v] of entries(x)) { + // color[v] = y[k]; + // } + // })() + // ) + // ) + // ); + + // return { + // obj: c2col, + // arr: c2col, + // str: c2hx, + // num: c2num, + // }[kind](); + + return color; } /**