Skip to content

Commit

Permalink
feat(parser): add support for import aliases (#13)
Browse files Browse the repository at this point in the history
  • Loading branch information
jonambas authored Apr 7, 2024
1 parent 2d1d973 commit d5513b6
Show file tree
Hide file tree
Showing 5 changed files with 138 additions and 50 deletions.
9 changes: 2 additions & 7 deletions src/__tests__/codegen.test.ts
Original file line number Diff line number Diff line change
@@ -1,10 +1,5 @@
import { codegen } from '../codegen';
import { createContext } from '../context';

const context = createContext({
foo: { 100: { value: '#fff' }, 200: { value: { base: '#000' } } },
bar: { 100: 'red', 200: 'blue' },
});
import { context } from './fixtures';

describe('codegen', () => {
it('generates ct runtime code', () => {
Expand Down Expand Up @@ -33,7 +28,7 @@ describe('codegen', () => {
"files": [
{
"code": "
const pluginCtMap = new Map([["foo.100","#fff"],["foo.200",{"base":"#000"}],["bar.100","red"],["bar.200","blue"]]);
const pluginCtMap = new Map([["foo.100","#fff"],["foo.200",{"base":"#000","lg":"#111"}],["bar.100","red"],["bar.200","blue"]]);
export const ct = (path) => {
if (!pluginCtMap.has(path)) return 'panda-plugin-ct_alias-not-found';
Expand Down
8 changes: 8 additions & 0 deletions src/__tests__/fixtures.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
import { createContext } from '../context';

export const tokens = {
foo: { 100: { value: '#fff' }, 200: { value: { base: '#000', lg: '#111' } } },
bar: { 100: 'red', 200: 'blue' },
};

export const context = createContext(tokens);
8 changes: 3 additions & 5 deletions src/__tests__/map.test.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,5 @@
import { get, makeMap, makePaths, mapTemplate } from '../map';

const tokens = {
foo: { 100: { value: '#fff' }, 200: { value: { base: '#000' } } },
bar: { 100: 'red', 200: 'blue' },
};
import { tokens } from './fixtures';

describe('get', () => {
it('gets a string', () => {
Expand All @@ -15,6 +11,7 @@ describe('get', () => {
`
{
"base": "#000",
"lg": "#111",
}
`,
);
Expand Down Expand Up @@ -67,6 +64,7 @@ describe('makeMap', () => {
"foo.100" => "#fff",
"foo.200" => {
"base": "#000",
"lg": "#111",
},
"bar.100" => "red",
"bar.200" => "blue",
Expand Down
120 changes: 98 additions & 22 deletions src/__tests__/parser.test.ts
Original file line number Diff line number Diff line change
@@ -1,37 +1,113 @@
import { parser } from '../parser';
import { createContext } from '../context';
import { context } from './fixtures';

const context = createContext({
foo: { 100: { value: '#fff' }, 200: { value: { base: '#000' } } },
bar: { 100: 'red', 200: 'blue' },
});
export const makeParser = (content: string) => {
return parser(
{
configure: () => {},
filePath: 'test.tsx',
content,
},
context,
);
};

describe('parser', () => {
it('parses', () => {
const res = parser(
{
configure: () => {},
filePath: 'test.tsx',
content: `<div className={css({ bg: ct("foo.200"), color: ct('bar.100'))})/>`,
const res = makeParser(`
import { css, ct, cva } from '@/styled-system/css';
const styles = cva({
base: {
// background: ct('foo.200'),
color: ct('bar.200'),
},
context,
);
});
export const Component = () => {
return (<div
className={
css({
bg: ct('foo.200'),
color: ct('bar.100')
})}
/>);
`);

expect(res).toMatchInlineSnapshot(
`"<div className={css({ bg: {"base":"#000"}, color: 'red')})/>"`,
`
"import { css, ct, cva } from '@/styled-system/css';
const styles = cva({
base: {
// background: ct('foo.200'),
color: 'blue',
},
});
export const Component = () => {
return (<div
className={
css({
bg: {"base":"#000","lg":"#111"},
color: 'red'
})}
/>);
"
`,
);
});

it('skips without "ct(" in contents', () => {
const res = parser(
{
configure: () => {},
filePath: 'test.tsx',
content: `<div className={css({ bg: "red.200" })/>`,
it('parses with an alias', () => {
const res = makeParser(`
import { css, ct as alias, cva } from '@/styled-system/css';
const styles = cva({
base: {
// background: ct('foo.200'),
color: ct('bar.200'),
},
context,
);
});
export const Component = () => {
return (<div
className={
css({
bg: alias('foo.200'),
color: alias('bar.100')
})}
/>);
`);

expect(res).toMatchInlineSnapshot(`
"import { css, ct as alias, cva } from '@/styled-system/css';
const styles = cva({
base: {
// background: ct('foo.200'),
color: ct('bar.200'),
},
});
export const Component = () => {
return (<div
className={
css({
bg: {"base":"#000","lg":"#111"},
color: 'red'
})}
/>);
"
`);
});

it('skips without ct imports or expressions', () => {
expect(
makeParser(`<div className={css({ bg: ct("foo.200") })/>`),
).toBeUndefined();

expect(res).toBeUndefined();
expect(
makeParser(`import { ct } from '@/styled-system/css`),
).toBeUndefined();
});
});
43 changes: 27 additions & 16 deletions src/parser.ts
Original file line number Diff line number Diff line change
@@ -1,36 +1,47 @@
import type { ParserResultBeforeHookArgs } from '@pandacss/types';
import type { PluginContext } from './types';
import { isObject } from './utils';
import { ts } from 'ts-morph';

export const parser = (
args: ParserResultBeforeHookArgs,
context: PluginContext,
): string | void => {
const { project, map } = context;

// TODO: handle `import { ct as xyz }` aliasing
const content = args.content;
if (!content.includes('ct(')) return;

const source = project.createSourceFile('__temp-ct-parser.ts', content, {
// Note: parser won't replace `ct` calls in JSX without .tsx
const source = project.createSourceFile('__ct-parser.tsx', args.content, {
overwrite: true,
});

let text = source.getText();
const calls = text.match(/ct\(['"][\w.]+['"]\)/g) ?? [];
let exists = false;
let alias = 'ct';

for (const node of source.getImportDeclarations()) {
if (!node.getText().includes('ct')) continue;
for (const named of node.getNamedImports()) {
if (named.getText() === 'ct' || named.getText().startsWith('ct as')) {
exists = true;
alias = named.getAliasNode()?.getText() ?? 'ct';
break;
}
}
}

if (!exists) return;

const calls = source
.getDescendantsOfKind(ts.SyntaxKind.CallExpression)
.filter((node) => node.getExpression().getText() === alias);

for (const call of calls) {
const path = call
.match(/['"][\w.]+['"]/)
?.toString()
.replace(/['"]/g, '');
if (!path) continue;
for (const node of calls) {
const path = node.getArguments()[0]?.getText().replace(/['"]/g, '');
const value = map.get(path);
text = text.replace(
call,

node.replaceWithText(
isObject(value) ? JSON.stringify(value) : `'${value}'`,
);
}

return text;
return calls.length ? source.getText() : undefined;
};

0 comments on commit d5513b6

Please sign in to comment.