diff --git a/.changeset/new-string-nodes.md b/.changeset/new-string-nodes.md new file mode 100644 index 000000000..b60222cef --- /dev/null +++ b/.changeset/new-string-nodes.md @@ -0,0 +1,8 @@ +--- +"@tokens-studio/graph-engine": minor +--- + +Added new string manipulation nodes: +- Case Convert: Transform strings between camelCase, snake_case, kebab-case, and PascalCase +- Replace: Simple string replacement without regex +- Normalize: String normalization with accent removal options \ No newline at end of file diff --git a/packages/documentation/docs/nodes/string/lowercase.mdx b/packages/documentation/docs/nodes/string/lowercase.mdx deleted file mode 100644 index 4de647693..000000000 --- a/packages/documentation/docs/nodes/string/lowercase.mdx +++ /dev/null @@ -1,62 +0,0 @@ ---- -title: Lowercase ---- -import { Editor, PanelGroup, DropPanelStore, PanelItem } from '@tokens-studio/graph-editor'; -import {InitialLayout} from '@site/src/components/editor/layout'; -import Tabs from '@theme/Tabs'; -import TabItem from '@theme/TabItem'; - - -# lowercase - -Node Type ID: `"studio.tokens.string.lowercase"` - -## Package - -This is a core package available at [Graph Engine](https://www.npmjs.com/package/@tokens-studio/graph-engine) - -To install - - - -```bash -npm install @tokens-studio/graph-engine -``` - - -```bash -yarn add @tokens-studio/graph-engine -``` - - - - - -## Description -Converts a string to lowercase - -## Inputs - - -#### value - -*No description* - - - - -## Outputs - - -#### value - -*No description* - - - - - -## Example - -*No examples* - diff --git a/packages/graph-engine/src/nodes/string/case.ts b/packages/graph-engine/src/nodes/string/case.ts new file mode 100644 index 000000000..71085c30e --- /dev/null +++ b/packages/graph-engine/src/nodes/string/case.ts @@ -0,0 +1,100 @@ +import { INodeDefinition, ToInput, ToOutput } from '../../index.js'; +import { Node } from '../../programmatic/node.js'; +import { StringSchema } from '../../schemas/index.js'; + +export enum CaseType { + CAMEL = 'camel', + SNAKE = 'snake', + KEBAB = 'kebab', + PASCAL = 'pascal' +} + +/** + * This node converts strings between different case formats + */ +export default class NodeDefinition extends Node { + static title = 'Case Convert'; + static type = 'studio.tokens.string.case'; + static description = 'Converts strings between different case formats'; + + declare inputs: ToInput<{ + string: string; + type: CaseType; + /** + * Characters to be replaced with spaces. Default: -_. + * @default "-_." + */ + delimiters: string; + }>; + declare outputs: ToOutput<{ + string: string; + }>; + + constructor(props: INodeDefinition) { + super(props); + this.addInput('string', { + type: StringSchema + }); + this.addInput('type', { + type: { + ...StringSchema, + enum: Object.values(CaseType), + default: CaseType.CAMEL + } + }); + this.addInput('delimiters', { + type: { + ...StringSchema, + default: '-_.' + } + }); + this.addOutput('string', { + type: StringSchema + }); + } + + execute(): void | Promise { + const { string, type, delimiters } = this.getAllInputs(); + + // Replace each delimiter with a space + const processedString = delimiters + .split('') + .reduce((result, char) => result.replaceAll(char, ' '), string); + + // First normalize the string by splitting on word boundaries + const words = processedString + // Add space before capitals in camelCase/PascalCase + .split(/([A-Z][a-z]+)/) + .join(' ') + // Remove extra spaces and convert to lowercase + .trim() + .toLowerCase() + // Split into words and remove empty strings + .split(/\s+/) + .filter(word => word.length > 0); + + let result: string; + switch (type) { + case CaseType.CAMEL: + result = words[0] + words.slice(1).map(capitalize).join(''); + break; + case CaseType.SNAKE: + result = words.join('_'); + break; + case CaseType.KEBAB: + result = words.join('-'); + break; + case CaseType.PASCAL: + result = words.map(capitalize).join(''); + break; + default: + result = string; + } + + this.outputs.string.set(result); + } +} + +function capitalize(str: string): string { + return str.charAt(0).toUpperCase() + str.slice(1); +} diff --git a/packages/graph-engine/src/nodes/string/index.ts b/packages/graph-engine/src/nodes/string/index.ts index 86910ed47..da8396e3f 100644 --- a/packages/graph-engine/src/nodes/string/index.ts +++ b/packages/graph-engine/src/nodes/string/index.ts @@ -1,8 +1,11 @@ +import caseConvert from './case.js'; import interpolate from './interpolate.js'; import join from './join.js'; import lowercase from './lowercase.js'; +import normalize from './normalize.js'; import pad from './pad.js'; import regex from './regex.js'; +import replace from './replace.js'; import split from './split.js'; import stringify from './stringify.js'; import uppercase from './uppercase.js'; @@ -10,9 +13,12 @@ import uppercase from './uppercase.js'; export const nodes = [ interpolate, join, + caseConvert, lowercase, + normalize, pad, regex, + replace, split, stringify, uppercase diff --git a/packages/graph-engine/src/nodes/string/normalize.ts b/packages/graph-engine/src/nodes/string/normalize.ts new file mode 100644 index 000000000..8c0b2fbef --- /dev/null +++ b/packages/graph-engine/src/nodes/string/normalize.ts @@ -0,0 +1,70 @@ +import { INodeDefinition, ToInput, ToOutput } from '../../index.js'; +import { Node } from '../../programmatic/node.js'; +import { StringSchema } from '../../schemas/index.js'; + +export enum NormalizationForm { + NFD = 'NFD', + NFC = 'NFC', + NFKD = 'NFKD', + NFKC = 'NFKC' +} + +/** + * This node normalizes strings and can remove diacritical marks (accents) + */ +export default class NodeDefinition extends Node { + static title = 'Normalize'; + static type = 'studio.tokens.string.normalize'; + static description = + 'Normalizes strings and optionally removes diacritical marks'; + + declare inputs: ToInput<{ + string: string; + form: NormalizationForm; + removeAccents: boolean; + }>; + declare outputs: ToOutput<{ + string: string; + }>; + + constructor(props: INodeDefinition) { + super(props); + this.addInput('string', { + type: StringSchema + }); + this.addInput('form', { + type: { + ...StringSchema, + enum: Object.values(NormalizationForm), + default: NormalizationForm.NFC + } + }); + this.addInput('removeAccents', { + type: { + type: 'boolean', + title: 'Remove Accents', + description: 'Whether to remove diacritical marks', + default: true + } + }); + this.addOutput('string', { + type: StringSchema + }); + } + + execute(): void | Promise { + const { string, form, removeAccents } = this.getAllInputs(); + + let result = string.normalize(form); + + if (removeAccents) { + // Remove combining diacritical marks + result = result + .normalize('NFD') + .replace(/[\u0300-\u036f]/g, '') + .normalize(form); + } + + this.outputs.string.set(result); + } +} diff --git a/packages/graph-engine/src/nodes/string/replace.ts b/packages/graph-engine/src/nodes/string/replace.ts new file mode 100644 index 000000000..84becbe23 --- /dev/null +++ b/packages/graph-engine/src/nodes/string/replace.ts @@ -0,0 +1,47 @@ +import { INodeDefinition, ToInput, ToOutput } from '../../index.js'; +import { Node } from '../../programmatic/node.js'; +import { StringSchema } from '../../schemas/index.js'; + +/** + * This node replaces all occurrences of a search string with a replacement string + */ +export default class NodeDefinition extends Node { + static title = 'Replace'; + static type = 'studio.tokens.string.replace'; + static description = + 'Replaces all occurrences of a search string with a replacement string'; + + declare inputs: ToInput<{ + string: string; + search: string; + replace: string; + }>; + declare outputs: ToOutput<{ + string: string; + }>; + + constructor(props: INodeDefinition) { + super(props); + this.addInput('string', { + type: StringSchema + }); + this.addInput('search', { + type: StringSchema + }); + this.addInput('replace', { + type: { + ...StringSchema, + default: '' + } + }); + this.addOutput('string', { + type: StringSchema + }); + } + + execute(): void | Promise { + const { string, search, replace } = this.getAllInputs(); + const result = string.split(search).join(replace); + this.outputs.string.set(result); + } +} diff --git a/packages/graph-engine/tests/suites/nodes/string/case.test.ts b/packages/graph-engine/tests/suites/nodes/string/case.test.ts new file mode 100644 index 000000000..7b281a101 --- /dev/null +++ b/packages/graph-engine/tests/suites/nodes/string/case.test.ts @@ -0,0 +1,82 @@ +import { Graph } from '../../../../src/graph/graph.js'; +import { describe, expect, test } from 'vitest'; +import Node, { CaseType } from '../../../../src/nodes/string/case.js'; + +describe('string/case', () => { + test('should convert to camelCase with default delimiters', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('hello-world.test_case'); + node.inputs.type.setValue(CaseType.CAMEL); + node.inputs.delimiters.setValue('-_.'); + + await node.execute(); + + expect(node.outputs.string.value).toBe('helloWorldTestCase'); + }); + + test('should convert to snake_case with custom delimiters', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('Hello@World#Test'); + node.inputs.type.setValue(CaseType.SNAKE); + node.inputs.delimiters.setValue('@#'); + + await node.execute(); + + expect(node.outputs.string.value).toBe('hello_world_test'); + }); + + test('should convert to kebab-case with single delimiter', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('HelloWorld+test'); + node.inputs.type.setValue(CaseType.KEBAB); + node.inputs.delimiters.setValue('+'); + + await node.execute(); + + expect(node.outputs.string.value).toBe('hello-world-test'); + }); + + test('should convert to PascalCase with multiple custom delimiters', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('hello|world$test%case'); + node.inputs.type.setValue(CaseType.PASCAL); + node.inputs.delimiters.setValue('|$%'); + + await node.execute(); + + expect(node.outputs.string.value).toBe('HelloWorldTestCase'); + }); + test('should handle empty delimiters string', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('hello-world_test'); + node.inputs.type.setValue(CaseType.CAMEL); + node.inputs.delimiters.setValue(''); + + await node.execute(); + + // Should only split on spaces and handle camelCase conversion + expect(node.outputs.string.value).toBe('hello-world_test'); + }); + test('should handle multiple occurrences of delimiters', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('hello--world..test__case.test-hello_world'); + node.inputs.type.setValue(CaseType.CAMEL); + node.inputs.delimiters.setValue('-_.'); + + await node.execute(); + + expect(node.outputs.string.value).toBe('helloWorldTestCaseTestHelloWorld'); + }); +}); diff --git a/packages/graph-engine/tests/suites/nodes/string/normalize.test.ts b/packages/graph-engine/tests/suites/nodes/string/normalize.test.ts new file mode 100644 index 000000000..2f80f9ed3 --- /dev/null +++ b/packages/graph-engine/tests/suites/nodes/string/normalize.test.ts @@ -0,0 +1,68 @@ +import { Graph } from '../../../../src/graph/graph.js'; +import { describe, expect, test } from 'vitest'; +import Node, { + NormalizationForm +} from '../../../../src/nodes/string/normalize.js'; + +describe('string/normalize', () => { + test('should remove accents when removeAccents is true', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('café résumé'); + node.inputs.removeAccents.setValue(true); + node.inputs.form.setValue(NormalizationForm.NFC); + + await node.execute(); + + expect(node.outputs.string.value).toBe('cafe resume'); + }); + + test('should preserve accents when removeAccents is false', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('café résumé'); + node.inputs.removeAccents.setValue(false); + node.inputs.form.setValue(NormalizationForm.NFC); + + await node.execute(); + + expect(node.outputs.string.value).toBe('café résumé'); + }); + + test('should handle different normalization forms', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + // Using a string with a precomposed character (é) vs decomposed (e + ´) + const precomposed = 'café'; // é is a single character + const decomposed = 'cafe\u0301'; // e and ´ are separate characters + + node.inputs.string.setValue(decomposed); + node.inputs.removeAccents.setValue(false); + + // Test NFC (precomposed) + node.inputs.form.setValue(NormalizationForm.NFC); + await node.execute(); + expect(node.outputs.string.value).toBe(precomposed); + + // Test NFD (decomposed) + node.inputs.form.setValue(NormalizationForm.NFD); + await node.execute(); + expect(node.outputs.string.value).toBe(decomposed); + }); + + test('should handle empty string', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue(''); + node.inputs.removeAccents.setValue(true); + node.inputs.form.setValue(NormalizationForm.NFC); + + await node.execute(); + + expect(node.outputs.string.value).toBe(''); + }); +}); diff --git a/packages/graph-engine/tests/suites/nodes/string/replace.test.ts b/packages/graph-engine/tests/suites/nodes/string/replace.test.ts new file mode 100644 index 000000000..1916e055d --- /dev/null +++ b/packages/graph-engine/tests/suites/nodes/string/replace.test.ts @@ -0,0 +1,57 @@ +import { Graph } from '../../../../src/graph/graph.js'; +import { describe, expect, test } from 'vitest'; +import Node from '../../../../src/nodes/string/replace.js'; + +describe('string/replace', () => { + test('should replace all occurrences of a string', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('hello hello world'); + node.inputs.search.setValue('hello'); + node.inputs.replace.setValue('hi'); + + await node.execute(); + + expect(node.outputs.string.value).toBe('hi hi world'); + }); + + test('should handle empty replacement string', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('hello world'); + node.inputs.search.setValue('hello '); + node.inputs.replace.setValue(''); + + await node.execute(); + + expect(node.outputs.string.value).toBe('world'); + }); + + test('should handle search string not found', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue('hello world'); + node.inputs.search.setValue('xyz'); + node.inputs.replace.setValue('abc'); + + await node.execute(); + + expect(node.outputs.string.value).toBe('hello world'); + }); + + test('should handle empty input string', async () => { + const graph = new Graph(); + const node = new Node({ graph }); + + node.inputs.string.setValue(''); + node.inputs.search.setValue('test'); + node.inputs.replace.setValue('replace'); + + await node.execute(); + + expect(node.outputs.string.value).toBe(''); + }); +});