Tree Shaking in our Design System Library

We are developing with the best effort our nice not-so little design component library. It is like the big ones, but smaller. When you do a common library for multiple projects you encounter many new opportunities to learn. You have to set up beta releases, you need to get guidelines in place and of course you need to optimize it.

The trigger for optimization

Every programmer probably knows, that premature optimization is best avoided. However, when you realize that on page load you just download all your icons in a JSON format, no matter if you need it or not, you know that you need to do something.

Looking into it also showed that this is caused by our commonly share design system. This is a much bigger issue, since all of our frontends will suffer from the same issue.

Original implementation

When migrating components into the new and fancy design system, we just copied the components we needed. The Icon components was fairly simple. It went something like this:

import Icons from 'icons.json';
const Icon = ({ name }) => 
    <span dangerouslySetHTML={ Icons[name] } /> 

You can see the problem right away. All the svg is stored in a single icons.json file. This is really flexible since you can control the icon with a prop, but also means you need to download all icons if you use any of them.

And we had many of them, exactly 600kB of them. In perspective that is 4x amount the recommended initial bundle size, and it is half the size our complete bundle.

Solution

After a bit of thinking I realized that we cannot keep the interface. There is a reason that when you import icons from other libraries it always goes like import { MyIcon } from 'good-icons.

So what I did was to scriptify the whole process and split up each icon into its own component. Having an icon as a standalone component that is not dependent on any crazy imports is important, since then it can be split up and pulled into bundles individually.

The end result was consisting of 4 parts:

The template file was relatively simple:

export const __SVG_COMPONENT_NAME__ = (props) => 
    <span dangerouslySetHTML=`__SVG_DATA__` ...props />;

This is of course oversimplified (and the dangerous part was removed later) but as you can see, I just wrote some incorrect tsx with some string placeholders.

Then in the node script I kinda do the following:

// To run this srcipt:
// node generateIcons.js iconSvgPath iconTsxPath iconTemplatePath.tsx
import { basename, join } from 'path';
import { promises as fs } from 'fs';

function toPascalCase(str) {
  return str
    .replace(/[-_](.)/g, (_, char) => char.toUpperCase())
    .replace(/^(.)/, char => char.toUpperCase());
}

// Get command-line arguments
const args = process.argv.slice(2);
const [dirPath, outPath, templateFilePath, suffix] = args;

if (!dirPath || !outPath || !templateFilePath) {
  console.error("Usage: node script.js <dirPath> <outPath> <templateFilePath> <suffix>");
  process.exit(1);
}

async function processFiles() {
  try {
    const template = await fs.readFile(templateFilePath, 'utf-8');
    const exports = [];

    const files = await fs.readdir(dirPath, { withFileTypes: true });
    for (const file of files) {
      if (file.isFile() && file.name.endsWith('.svg')) {
        const filePath = join(dirPath, file.name);
        const svgContent = await fs.readFile(filePath, 'utf-8');

        const svgName = basename(file.name, '.svg');
        const pascalCaseName = toPascalCase(svgName) + suffix;

        const resultContent = template
          .replace(/__SVG_NAME__/g, svgName)
          .replace(/__SVG_COMPONENT_NAME__/g, pascalCaseName)
          .replace(/__SVG_DATA__/g, svgContent);

        const outputFilePath = join(outPath, `${pascalCaseName}.tsx`);
        await fs.writeFile(outputFilePath, resultContent);
        console.log(`Written to file: ${outputFilePath}`);

        exports.push(`export { ${pascalCaseName} } from './${pascalCaseName}';`);
      }
    }

    // Write all exports to index.ts
    const indexPath = join(outPath, 'index.ts');
    await fs.writeFile(indexPath, exports.sort().join('\n'));
    console.log(`All exports written to index.ts`);

  } catch (error) {
    console.error("An error occurred:", error);
  }
}

processFiles();

In short: - Iterate through the svg folder and for each icon - create a new file using the Template and replace strings like __SVG_NAME__ with correct values generating valid tsx files. - Create an index.ts file which exports all the generated components

The arguments are there because we started to use this script to generate the Illustration components too.

Summary

Is this the best possible solution? Probably not, but for the project size and the allocated time this gets the work done and fixing our bundle size issue.