三年前写的一个代码工具,至今已被 AI Coding 完全取代。

初衷

2022 年在 momenta 实习的时候就经常发现一个代码问题,大家编写的代码中经常出现 style props 和 tailwind css 并存的现象,就像是:

jsx 复制代码
<div className="inline-block" style={{ width: 300 }} />

这个简单的代码已经有了两个问题:

  1. 代码风格不统一。已经引入了 tailwind css,又使用 style props 仅仅作为样式而不是变量
  2. 总是引发重新渲染。style props 使用的字面量 object { width: 300 }, 会在每次父组件重新渲染的时候引发这个组件的重新渲染。如下面这个代码所示:
tsx 复制代码
const CompA = () => {
    const [count, setCount] = useState(0);
    
    return (
        <>
            <div> {count} </div>
            <button onClick={() => setCount(count + 1)}> +1 </button>
            
            <div style={{ width: 300 }}>
                {/* 这里是个巨大的组件 */}
            </div>
        </>
    )
}

编写一个 demo,左边是直接使用 object 字面量编写的 react 组件,右边是经过 useMemo 固化的 object字面量,其在点击按钮加一引发重新渲染后,左侧会跟个重新渲染,而右侧则保持静态不重新渲染。运行效果如图所示

这仅是一个演示 demo,当真正进入生产环境中,被牵连的组件可能规模比较庞大,重新渲染可能导致显示页面跳动、白屏,以及普遍的性能损失和性能优化失败

解决办法

这个问题解决办法的文字描述如下:

首先,行 AST 分析,得到每个组件的抽象语法树数据。

第二,遍历语法树,查询每个 Node 上是否有 style object。

第三,对 Style Object 中的 Properties 进行遍历并通过某些手段映射到 Tailwind CSS string 上。

第四,重构 AST,删除 Style object, 增加 className。

最后,保存文件。


上述步骤中最复杂的是第三步,其需要考虑多种情况:

  • 静态映射
  • 动态映射
  • 状态传递

对于静态映射,其变化为:

tsx 复制代码
{ textAlign: "left" }  -> "text-left"

对于动态映射,其变化为:

tsx 复制代码
{ marginLeft: 1 } or { marginLeft: '1px' } -> "ml-[1px]"

对于状态传递,其应该保留,一般展现形式为变量或者插值:

tsx 复制代码
{ position: 'absolute', top: `${topVar}` } -> "absolute" { top: `${topVar}` }

所以在当时的测试样例中我是这样写的:

js 复制代码
const testSamples = [
    {
        input: `<div style={{ textAlign: 'center' }} />;`,
        output: `className="text-center"`
    },
    {
        input: `<div style={{ textAlign: false }} />;`,
        output: `textAlign: false`
    },
    {
        input: `<div style={{ backgroudUnknow: 'black', textAlign: 'center'  }} />;`,
        output: `className="text-center"`
    },
    {
        input: `<div style={{ padding: '2px' }} />;`,
        output: `className="p-[2px]"`
    },
    {
        input: `<div style={{ padding: 2 }} />;`,
        output: `className="p-[2px]"`
    },
    {
        input: `<div style={{ padding: '2px 4px 2px' }} />;`,
        output: `className="pt-[2px] py-[4px] pb-[2px]"`
    },
    {
        input: `<div style={{ strokeWidth: 4 }} />`,
        output: `className="stroke-width-4"`
    },
    {
        input: `<div style={{ strokeWidth: '4' }} />`,
        output: `className="stroke-width-4"`
    },
    {
        input: `<div style={{ textAlign: 'center !important' }} />`,
        output: `className="!text-center"`
    },
    {
        input: `<div style={{ padding: '100%', margin: 'calc(100%)' }} />`,
        output: `className="p-[100%] m-[calc(100%)]"`
    },
    {
        input: `<div style={{ fontWeight: 500 }} />`,
        output: `className="font-500"`
    },
];

如今

如今,三年后各种 AI coding 工具层出不穷,Claude Code, Gemini Cli, 以及国内的 Qwen Coder,都可以仅用一句提示词做到上述的所有工作:

复制代码
帮我查看一下这个项目,并将所有组件中使用的 React Style Props 切换为 Tailwind css,并输出这样做的好处。

根据你的项目大小,也许 5-15 分钟后,全部的代码已经被改造完成了,并且 AI 的改动可能更具有逻辑性。

并且在你使用 AI 编写的代码中,很少会出现是使用 React Style Props 的情况了。

我的这个小项目也被正式的完全替代了。

相关推荐
Lee川12 小时前
Milvus 实战:当 RAG 遇上向量数据库,从"玩具 Demo"到"生产可用的"那一步
前端·数据库·人工智能
anOnion13 小时前
构建无障碍组件之Toolbar Pattern
前端·html·交互设计
惊鸿一博13 小时前
图标加载方式_zeroIcon_是否加前缀mdi
开发语言·前端·javascript
2501_9400417413 小时前
前端工程化进阶:5个高交互与可视化项目提示词
前端
你很易烊千玺13 小时前
JS 异步 从零讲(大白话 + 真实场景 + 可运行案例)
前端·javascript·vue.js
人月神话-Lee14 小时前
【图像处理】Sobel 边缘检测——让机器“看见“轮廓
图像处理·人工智能·计算机视觉·ios·ai编程·swift
孟健15 小时前
Hermes 升级后,我的 Telegram 附件突然发不出来了
ai编程
why技术15 小时前
AI Coding开始进入第四个时代,我还没上车呢!
前端·人工智能·后端
大家的林语冰16 小时前
CSS 已死?DOM 性能黑洞!Pretext 排版革命让你在文本间跳舞,没有 DOM 也能纵享丝滑~
前端·javascript·css
vipbic16 小时前
我也该升级了,陪伴了我7年的博客
前端