Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

docs: generate components with complex children for patternhub copy-code #2757

Merged
merged 25 commits into from
Jul 8, 2024
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
bff1cfa
docs: generate components with complex children for patternhub copy-code
nmerget Jun 18, 2024
0f7d8c7
Merge branch 'main' of github.com:db-ui/mono into docs-generate-copy-…
nmerget Jun 18, 2024
d67aa6e
fix: make exampleProps optional
nmerget Jun 18, 2024
db9cdb8
Update showcases/shared/tag.json
nmerget Jun 19, 2024
241f090
Update showcases/shared/tag.json
nmerget Jun 19, 2024
cefab18
Update showcases/shared/notification.json
nmerget Jun 19, 2024
ad8bee4
Update showcases/shared/notification.json
nmerget Jun 19, 2024
fef57c3
Update showcases/shared/notification.json
nmerget Jun 19, 2024
3ea7f4d
chore: update specs for accordion showcase
nmerget Jun 19, 2024
f433829
Merge remote-tracking branch 'origin/docs-generate-copy-code' into do…
nmerget Jun 19, 2024
1778346
fix: issue with initialSelectedIndex for Tabs
nmerget Jun 19, 2024
8e9e9c6
Merge branch 'main' into docs-generate-copy-code
mfranzke Jun 19, 2024
77df4c9
Merge branch 'main' into docs-generate-copy-code
nmerget Jun 19, 2024
24bfd1c
Merge branch 'main' of github.com:db-ui/mono into docs-generate-copy-…
nmerget Jun 24, 2024
19baf1b
Merge branch 'main' into docs-generate-copy-code
nmerget Jun 25, 2024
015e5f0
fix: linting issue
nmerget Jun 25, 2024
4c7b979
Merge branch 'main' into docs-generate-copy-code
nmerget Jun 27, 2024
d61f6c3
Merge branch 'main' into docs-generate-copy-code
nmerget Jul 5, 2024
bb72e55
Merge branch 'main' into docs-generate-copy-code
mfranzke Jul 8, 2024
36213d2
Update showcases/patternhub/scripts/utils.js
nmerget Jul 8, 2024
5282919
Update showcases/patternhub/scripts/utils.js
nmerget Jul 8, 2024
ea54731
Update packages/components/src/components/tabs/tabs.lite.tsx
nmerget Jul 8, 2024
3d5b199
Merge branch 'main' into docs-generate-copy-code
nmerget Jul 8, 2024
dbadfa6
Merge branch 'main' into docs-generate-copy-code
mfranzke Jul 8, 2024
c17c20d
Merge branch 'main' into docs-generate-copy-code
nmerget Jul 8, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 0 additions & 2 deletions output/vue/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,6 @@ Import the styles in scss or css. Based on your technology the file names could
```ts
// main.ts
import "./style.scss";
import "@db-ui/v-components/dist/style.css";
```

</details>
Expand All @@ -46,7 +45,6 @@ import "@db-ui/v-components/dist/style.css";
```ts
// main.ts
import "@db-ui/components/build/styles/db-ui-42-rollup.css";
import "@db-ui/v-components/dist/style.css";
```

</details>
Expand Down
2 changes: 1 addition & 1 deletion packages/components/src/components/popover/docs/React.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ For general installation and configuration take a look at the [react-components]
import { DBPopover, DBButton } from "@db-ui/react-components";

const App = () => (
<DBPopover slotTrigger={<DBButton>Hover on me to open Popover</DBButton>}>
<DBPopover trigger={<DBButton>Hover on me to open Popover</DBButton>}>
Use any html code here like e.g. a <code>button</code>:
<button type="button">Test</button>
</DBPopover>
Expand Down
4 changes: 3 additions & 1 deletion showcases/patternhub/scripts/generate-example-jsx.js
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,9 @@ const generateExampleJSX = () => {
const code = getCodeByFramework(
componentName,
'react',
example
example,
true,
variant.children
);
examples.push(
`"${componentName}${variant.name}${
Expand Down
11 changes: 6 additions & 5 deletions showcases/patternhub/scripts/get-code-files.js
Original file line number Diff line number Diff line change
@@ -1,15 +1,13 @@
/* eslint-disable no-await-in-loop */
import FS from 'node:fs';
import prettier from 'prettier';
import prettier0 from 'prettier/parser-babel.js';
import { allExamples } from './generated/index.jsx';
import { getCodeByFramework } from './utils.js';

const sharedPath = '../shared';
const reactPath = '../react-showcase/src/components';

const codeFrameworks = ['angular', 'html', 'react', 'vue'];
const plugins = [prettier0];

const getFileTypeByFramework = (framework) => {
if (framework === 'react') {
Expand All @@ -25,6 +23,7 @@ const getFileTypeByFramework = (framework) => {

const getExamplesAsMDX = async (componentName, variant) => {
const examples = variant.examples;
const children = variant.children;

let result =
"import { useEffect, useState } from 'react';\n" +
Expand Down Expand Up @@ -71,17 +70,19 @@ const getExamplesAsMDX = async (componentName, variant) => {
exampleCode = getCodeByFramework(
componentName,
framework,
example
example,
false,
children
);
}

try {
exampleCode = await prettier.format(exampleCode, {
parser: 'babel',
plugins
parser: framework === 'react' ? 'babel' : framework
});
} catch {
// We do not care about errors here
// console.error(e);
}

exampleCode = exampleCode?.replace(/;/g, '').trim();
Expand Down
186 changes: 153 additions & 33 deletions showcases/patternhub/scripts/utils.js
Original file line number Diff line number Diff line change
Expand Up @@ -19,55 +19,62 @@
};

/**
* @param componentName {string}
* @param props {object}
* @param framework {'angular'|'react'|'vue'}
* @param example {{name:string, props: object}}
* @returns {string}
* @param noEvents {boolean}
* @return {*[]}
*/
export const getCodeByFramework = (componentName, framework, example) => {
const properties = example.props;
let tag = `DB${componentName
.split('-')
.map((part) => part.charAt(0).toUpperCase() + part.slice(1))
.join('')}`;
if (framework === 'angular') {
tag = `db-${componentName}`;
}

const getAttributes = (props, framework, noEvents) => {
const attributes = [];

const propertyKeys = properties
? Object.keys(properties).filter((key) => key !== 'children')
// Some slots which shouldn't be attributes
const propertyKeys = props
? Object.keys(props).filter(
(key) =>
key !== 'children' &&
key !== 'component' &&
key !== 'identifier' &&
key !== 'img' &&
key !== 'link' &&
key !== 'noContent'
)
: [];

for (const key of propertyKeys) {
let value = properties[key];
let value = props[key];
const isEventListener = key.startsWith('on');

if (value instanceof Object) {
value = JSON.stringify(value);
if (noEvents && (isEventListener || value === key)) {
continue;
}

if (
typeof properties[key] === 'boolean' ||
typeof properties[key] === 'number' ||
properties[key] instanceof Object ||
key === 'click'
typeof value === 'boolean' ||
typeof value === 'number' ||
nmerget marked this conversation as resolved.
Show resolved Hide resolved
value instanceof Object ||
value === key ||
isEventListener
) {
if (
framework !== 'react' &&
(properties[key] instanceof Object || key === 'click')
) {
value = value.replaceAll('"', "'");
if (value instanceof Object) {
value = JSON.stringify(value);
}

if (framework === 'angular') {
attributes.push(`[${key}]="${value}"`);
if (isEventListener) {
attributes.push(`(${key})="${value}"`);
} else {
attributes.push(`[${key}]="${value}"`);
}
} else if (framework === 'vue') {
attributes.push(`:${key}="${value}"`);
} else if (framework === 'react' && key === 'click') {
attributes.push(`onClick={${value}}`);
} else if (typeof properties[key] === 'boolean') {
if (isEventListener) {
attributes.push(`@${key}="${value}"`);
} else {
attributes.push(`:${key}="${value}"`);
}
} else if (typeof props[key] === 'boolean') {
attributes.push(key);
} else if (isEventListener) {
attributes.push(`${key}={()=>${value}}`);
Dismissed Show dismissed Hide dismissed
} else {
attributes.push(`${key}={${value}}`);
}
Expand All @@ -76,7 +83,120 @@
}
}

return `<${tag} ${attributes.join(' ')}>${example.name}</${tag}>`;
return attributes;
};

const getTag = (componentName) =>
componentName
.split('-')
.map((part) => part.charAt(0).toUpperCase() + part.slice(1))
.join('');

/**
* @param componentName {string}
* @param framework {'angular'|'react'|'vue'}
* @param example {{name:string, props: object, className?:string, content?:string,children?:{name:string, props: object}[]}}
* @param noEvents {boolean}
* @param [children] {{name:string, props: object,slot?:string, angularDirective?:boolean, content?:string,children?:{name:string, props: object}[]}[]}
* @returns {string}
*/
export const getCodeByFramework = (
componentName,
framework,
example,
noEvents,
children
) => {
const { props, name, content } = example;
let className = '';
let tag = `DB${getTag(componentName)}`;
if (framework === 'angular') {
tag = `db-${componentName}`;
}

if (componentName === 'img' || componentName === 'a') {
nmerget marked this conversation as resolved.
Show resolved Hide resolved
tag = componentName;
}

if (example.className) {
className =
framework === 'react'
? ` className="${example.className}"`
: ` class="${example.className}"`;
}

const attributes = getAttributes(props, framework, noEvents);
const nonSlots = (children ?? example.children)?.filter(
(child) =>
!child.slot ||
(child.slot.includes('Navigation') && framework !== 'angular')
);
const innerContent =
nonSlots?.length > 0
? nonSlots
.map((child) =>
getCodeByFramework(
child.name,
framework,
child,
noEvents,
child.children
)
)
.join('\n') + (content ?? '')
: content ?? name;

const slots = (children ?? example.children)?.filter((child) =>
child.slot
? !(child.slot.includes('Navigation') && framework !== 'angular')
: false
);
let reactSlots = '';
let otherSlots = '';
if (slots) {
if (framework === 'react') {
reactSlots =
' ' +
slots
.map((child) => {
let slotName = getTag(child.slot);
slotName =
slotName.charAt(0).toLowerCase() +
slotName.slice(1);
return `${slotName}={${getCodeByFramework(
child.name,
framework,
child,
noEvents,
child.children
)}}`;
})
.join('\n');
} else {
otherSlots =
' ' +
slots
.map((child) => {
const resolvedSlot = getCodeByFramework(
child.name,
framework,
child,
noEvents,
child.children
);
if (framework === 'angular') {
return `<ng-container ${child.angularDirective ? `*db${getTag(child.slot)}` : child.slot}>${resolvedSlot}</ng-container>`;
}

return `<template v-slot:${child.slot}>${resolvedSlot}</template>`;
})
.join('\n');
}
}

return `<${tag}${className} ${attributes.join(' ')}${reactSlots}>
${otherSlots}${innerContent}
</${tag}>`;
};

export const getColorVariants = () => [
Expand Down
Loading
Loading