diff --git a/packages/design-system/src/directives/index.ts b/packages/design-system/src/directives/index.ts new file mode 100644 index 0000000000000..eb939f0531e3e --- /dev/null +++ b/packages/design-system/src/directives/index.ts @@ -0,0 +1 @@ +export { n8nTruncate } from './n8n-truncate'; diff --git a/packages/design-system/src/directives/n8n-truncate.test.ts b/packages/design-system/src/directives/n8n-truncate.test.ts new file mode 100644 index 0000000000000..ec8922d2bc76b --- /dev/null +++ b/packages/design-system/src/directives/n8n-truncate.test.ts @@ -0,0 +1,76 @@ +import { render } from '@testing-library/vue'; +import { n8nTruncate } from './n8n-truncate'; + +describe('Directive n8n-truncate', () => { + it('should truncate text to 30 chars by default', async () => { + const { html } = render( + { + props: { + text: { + type: String, + }, + }, + template: '
{{text}}
', + }, + { + props: { + text: 'This is a very long text that should be truncated', + }, + global: { + directives: { + n8nTruncate, + }, + }, + }, + ); + expect(html()).toBe('
This is a very long text that...
'); + }); + + it('should truncate text to 30 chars in case of wrong argument', async () => { + const { html } = render( + { + props: { + text: { + type: String, + }, + }, + template: '
{{text}}
', + }, + { + props: { + text: 'This is a very long text that should be truncated', + }, + global: { + directives: { + n8nTruncate, + }, + }, + }, + ); + expect(html()).toBe('
This is a very long text that...
'); + }); + + it('should truncate text to given length', async () => { + const { html } = render( + { + props: { + text: { + type: String, + }, + }, + template: '
{{text}}
', + }, + { + props: { + text: 'This is a very long text that should be truncated', + }, + global: { + directives: { + n8nTruncate, + }, + }, + }, + ); + expect(html()).toBe('
This is a very long text...
'); + }); +}); diff --git a/packages/design-system/src/directives/n8n-truncate.ts b/packages/design-system/src/directives/n8n-truncate.ts new file mode 100644 index 0000000000000..8d454654c07d3 --- /dev/null +++ b/packages/design-system/src/directives/n8n-truncate.ts @@ -0,0 +1,26 @@ +import type { DirectiveBinding, ObjectDirective } from 'vue'; +import { truncate } from '../utils/string'; + +/** + * Custom directive `n8nTruncate` to truncate text content of an HTML element. + * + * Usage: + * In your Vue template, use the directive `v-n8n-truncate` with an argument to specify the length to truncate to. + * + * Example: + *

Some long text that will be truncated

+ * + * This will truncate the text content of the paragraph to 10 characters. + * + * Hint: Do not use it on components + * https://vuejs.org/guide/reusability/custom-directives#usage-on-components + */ + +export const n8nTruncate: ObjectDirective = { + mounted(el: HTMLElement, binding: DirectiveBinding) { + el.textContent = truncate(el.textContent ?? '', Number(binding.arg) || undefined); + }, + updated(el: HTMLElement, binding: DirectiveBinding) { + el.textContent = truncate(el.textContent ?? '', Number(binding.arg) || undefined); + }, +}; diff --git a/packages/design-system/src/main.ts b/packages/design-system/src/main.ts index 54934d6f1c4d6..dddb89f888ff4 100644 --- a/packages/design-system/src/main.ts +++ b/packages/design-system/src/main.ts @@ -5,4 +5,5 @@ export * from './components'; export * from './plugin'; export * from './types'; export * from './utils'; +export * from './directives'; export { locale }; diff --git a/packages/design-system/src/plugin.ts b/packages/design-system/src/plugin.ts index 493fde38ecc5a..c6d3d2a6c9f30 100644 --- a/packages/design-system/src/plugin.ts +++ b/packages/design-system/src/plugin.ts @@ -1,5 +1,6 @@ import type { Component, Plugin } from 'vue'; import * as components from './components'; +import * as directives from './directives'; export interface N8nPluginOptions {} @@ -8,5 +9,9 @@ export const N8nPlugin: Plugin = { for (const [name, component] of Object.entries(components)) { app.component(name, component as unknown as Component); } + + for (const [name, directive] of Object.entries(directives)) { + app.directive(name, directive); + } }, }; diff --git a/packages/design-system/src/utils/string.test.ts b/packages/design-system/src/utils/string.test.ts new file mode 100644 index 0000000000000..6f65775f9ac67 --- /dev/null +++ b/packages/design-system/src/utils/string.test.ts @@ -0,0 +1,17 @@ +import { truncate } from './string'; + +describe('Utils string', () => { + describe('truncate', () => { + it('should truncate text to 30 chars by default', () => { + expect(truncate('This is a very long text that should be truncated')).toBe( + 'This is a very long text that...', + ); + }); + + it('should truncate text to given length', () => { + expect(truncate('This is a very long text that should be truncated', 25)).toBe( + 'This is a very long text...', + ); + }); + }); +}); diff --git a/packages/design-system/src/utils/string.ts b/packages/design-system/src/utils/string.ts new file mode 100644 index 0000000000000..9170b57c00733 --- /dev/null +++ b/packages/design-system/src/utils/string.ts @@ -0,0 +1,2 @@ +export const truncate = (text: string, length = 30): string => + text.length > length ? text.slice(0, length).trim() + '...' : text;