Vite+Antd+Micro-app中iframe模式下样式闪烁的问题

Vite中使用Micro-app进行微前端开发,做的过程中发现在iframe模式下,无论是首屏加载还是中间的路由切换都会出现antd样式闪烁;具体如下:

  1. 按钮的样式从最初始状态变为AntdButton样式,有颜色改变时,最为明显
  2. 输入框由input初始样式变为Antd的样式
  3. 所有的Antd组件都是从浏览器的默认样式加载为Antd的样式

解决历程

  1. 既然是样式后加载,那么是否可以进行样式预设,当Antd样式加载时,就不会出现明显的闪烁
js 复制代码
// index.html
<!-- iframe 模式:预设关键样式,避免 FOUC -->
    <style>
      /* 预设主题色,避免按钮颜色闪烁 */
      .ant-btn-primary {
        background-color: #7470e9;
        border-color: #7470e9;
        color: #fff;
      }

      .ant-input {
        height: 40px;
        font-size: 16px;
        padding: 6.5px 11px;
        border-width: 1px;
        border-style: solid;
        border-color: #d9d9d9;
        border-radius: 6px;
      }

      .ant-btn {
        border-width: 1px;
      }
    </style>

解决效果:肉眼可见的闪烁没有了,但是缺点也很明显,就是需要把用到的所有的Antd组件的基础样式或者叫引起明显抖动的样式预设,避免明显抖动

  1. 仿照AntdNext.js做的@ant-design/nextjs-registry库,来解决提前加载样式的问题

@ant-design/nextjs-registry是如何解决在Next.js中首屏加载时闪烁的问题,是在Next.js渲染组件时,使用@ant-design/cssinjs提供的const styles = extractStyle(cache, true)能力

js 复制代码
'use client'
const AntdRegistry = ({ children }) => {
  useServerInsertedHTML(() => {
    // 在服务端渲染时执行
    const styleText = extractStyle(cache, true);
    return <style dangerouslySetInnerHTML={{ __html: styleText }} />;
  });

  return <StyleProvider cache={cache}>{children}</StyleProvider>;
};

将含AntdHTML代码全部发给浏览器,这样渲染的时候就不会出现样式闪烁,但是 这个是Next.js专属能力CSR(客户端渲染)的做不到,因为Antd的组件样式是在加载组件时才会放到cache中,也就是说在App.tsx中使用下面的代码包括,是没有用的

js 复制代码
useEffect(() => {
    // 在客户端挂载时执行
    const styleText = extractStyle(cache, true);
    const styleElement = document.createElement('style');
    styleElement.textContent = styleText;
    document.head.insertBefore(styleElement, document.head.firstChild);
  }, []);

这个代码在执行时styleText啥也没有,加载组件在useEffect之后,此时我们已经很接近真相了

  1. 我发现一个现象,在SPA页面中,样式只会在首屏时闪烁,进入其他页面后,并没有闪烁,已加载的样式被缓存了,查询之后,发现是Antdcache在做缓存判断,然后我查看了子应用的样式发现Micro-app给每个样式否加了前缀micro-app[name=hospital-gateway]
css 复制代码
micro-app[name=hospital-gateway] ._hospital-search-wrapper_v077x_1 ._search-operate_v077x_5 button {
    width: 100px;
    margin-left: 8px;
}

这个前缀导致了Antdcache失效,而且每次路由切换时,Micro-app都会将Antd原来的样式重写为有micro-app[name=hospital-gateway]前缀的样式,这就是导致每次样式抖动的根本原因。

  1. 找到根源后,就看如何去掉这个前缀,iframe模式下,样式是天然隔离的,是不需要类似with模式下的样式前缀的,而且去掉前缀,也可以启用Antd的样式缓存,避免切换路由时的样式抖动,最后发现在添加disable-scopecss属性就可以去掉自动添加的前缀
js 复制代码
<micro-app name={name} url={url} baseroute keep-alive disable-scopecss />
相关推荐
方也_arkling36 分钟前
别名路径联想提示。@/统一文件路径的配置
前端·javascript
毕设源码-朱学姐38 分钟前
【开题答辩全过程】以 基于web教师继续教育系统的设计与实现为例,包含答辩的问题和答案
前端
web打印社区1 小时前
web-print-pdf:突破浏览器限制,实现专业级Web静默打印
前端·javascript·vue.js·electron·html
RFCEO1 小时前
前端编程 课程十三、:CSS核心基础1:CSS选择器
前端·css·css基础选择器详细教程·css类选择器使用方法·css类选择器命名规范·css后代选择器·精准选中嵌套元素
Amumu121382 小时前
Vuex介绍
前端·javascript·vue.js
We་ct2 小时前
LeetCode 54. 螺旋矩阵:两种解法吃透顺时针遍历逻辑
前端·算法·leetcode·矩阵·typescript
2601_949480062 小时前
【无标题】
开发语言·前端·javascript
css趣多多2 小时前
Vue过滤器
前端·javascript·vue.js
理人综艺好会3 小时前
Web学习之用户认证
前端·学习
We་ct3 小时前
LeetCode 36. 有效的数独:Set实现哈希表最优解
前端·算法·leetcode·typescript·散列表