Nuxt 团队与 Chrome Aurora 团队在 Google 的合作下,兴奋地宣布 Nuxt Scripts 的公开测试版发布。
Nuxt Scripts 是处理第三方脚本的更好方式,提供了更好的性能、隐私、安全性和开发者体验。
开始使用 Nuxt Scripts
在一年前,Daniel 发布了初步的 Nuxt Scripts RFC。该 RFC 提出了一个模块,旨在“允许管理和优化第三方脚本,遵循性能和合规网站的最佳实践”。
拥有 个人经验 解决与第三方脚本相关的性能问题,我深知这些性能优化会多么困难。尽管如此,我仍然渴望迎接这个挑战,并接手了项目。
以 RFC 为理念的种子,我开始原型设计其可能的 样子,并使用 Unhead 进行开发。
在思考我希望确切构建的内容时,我发现真正的问题不仅是如何加载“优化过的”第三方脚本,而是如何整体上改善与第三方脚本的合作体验。
为什么要构建第三方脚本模块?
94%的网站至少使用一个第三方提供商,平均每个网站有 五个第三方提供商。
我们知道第三方脚本并不完美;它们 拖慢网页速度,造成隐私和安全问题,并且与之合作非常麻烦。
然而,它们在根本上是有用的,短期内不会消失。
通过探索第三方脚本的问题,我们可以看到可以进行改进的地方。
😒 开发者体验:全栈的头痛
让我们通过向您的 Nuxt 应用添加一个虚构的 tracker.js
脚本,该脚本向窗口添加 track
函数,来了解一下。
我们开始使用 useHead
加载脚本。
useHead({ scripts: [{ src: '/tracker.js', defer: true }] })
但是,现在我们尝试在应用中使脚本功能正常工作。
在 Nuxt 中处理第三方脚本时,以下步骤是常见的:
- 一切必须包装以确保 SSR 安全。
- 不可靠的检查以确认脚本是否已加载。
- 为类型增强窗口对象。
<script setup>
// ❌ 哦不,window 未定义!
// 💡 如果我们在 Nuxt 中使用 SSR,窗口无法直接访问。
// 👉 我们需要确保这个 SSR 是安全的
window.track('page_view', useRoute().path)
</script>
🐌 性能:“为什么我无法在 Lighthouse 上获得 100?”
为了使访问者能够开始与您的 Nuxt 网站互动,应用程序包需要被下载,Vue 需要对应用实例进行水合。
加载第三方脚本可能会干扰此水合过程,即使使用 async
或 defer
。这会减慢网络速度并阻塞主线程,从而导致用户体验下降和糟糕的 核心网络指标。
Chrome 用户体验报告 显示,使用许多第三方资源的 Nuxt 网站通常在 交互到下一个绘制 (INP) 和 最大内容绘制 (LCP) 分数上较低。
为了查看第三方脚本如何降低性能,我们可以查看 Web Almanac 2022。报告显示,前10个第三方脚本 平均中断时间为 1.4 秒。
🛡️ 隐私与安全:不做坏事?
在前 10,000 个网站中,有58%网站存在 在外部 cookie 中交换跟踪 ID 的第三方脚本,这意味着它们可以在用户禁用第三方 cookie 的情况下跨网站跟踪用户。
虽然在许多情况下,我们与使用的提供商的选择受到限制,但我们应该尽量减少泄露终端用户数据的数量。
当我们涉及隐私问题时,准确传达这些信息也可能很困难,并且需要构建所需的同意管理,以遵守 GDPR 等法规。
使用第三方脚本时安全性也是一个关注点。第三方脚本是恶意行为者常用的攻击向量,大多数并未为其脚本提供 integrity
哈希,这意味着它们可以随时被破坏,并将恶意代码注入到您的应用程序中。
Nuxt Scripts 如何解决这些问题?
可组合的: useScript
这个可组合函数位于 <script>
标签和添加到 window.{thirdPartyKey}
的功能之间。
对于 <script>
标签,可组合函数:
- 完全显示脚本的加载和错误状态
- 默认情况下,在 Nuxt 水合应用时加载脚本,以获得稍微更好的性能。
- 限制
crossorigin
和referrerpolicy
以提高隐私和安全性。 - 提供方法以 延迟加载脚本,直到需要为止。
对于脚本 API,它:
- 提供完全类型安全的脚本功能
- 添加一个代理层,允许你的应用在脚本功能处于不安全的上下文中运行(SSR、脚本未加载前、脚本被阻止)
const { proxy, onLoaded } = useScript('/hello.js', {
trigger: 'onNuxtReady',
use() {
return window.helloWorld
}
})
onLoaded(({ greeting }) => {
// ✅ 脚本已加载!挂钩到 Vue 生命周期
})
// ✅ 或者使用代理 API - 友好 SSR,在脚本加载时调用
proxy.greeting() // Hello, World!
declare global {
interface Window {
helloWorld: {
greeting: () => 'Hello World!'
}
}
}
脚本注册表
脚本注册表 是一个常见第三方脚本的一系列第一方集成。发布时,我们支持 21 个脚本,未来还会有更多。
这些注册表脚本是围绕 useScript
进行微调的包装,具有完整的类型安全性,运行时验证脚本选项(仅限开发)和环境变量支持。
例如,我们可以查看 Fathom Analytics 脚本。
const { proxy } = useScriptFathomAnalytics({
// ✅ 选项在运行时被验证
site: undefined
})
// ✅ 类型安全
proxy.trackPageview()
外观组件
注册表包含几个 外观组件,如 Google Maps、YouTube 和 Intercom。
外观组件是“虚假”组件,当第三方脚本加载时进行水合。 外观组件有权衡,但可以大幅提高您的性能。请参阅 什么是外观组件?, 了解有关更多信息,。
Nuxt Scripts 提供外观组件作为可访问但无头的组件,意味着它们默认不带样式, 但添加必要的可访问性数据。
Click to load
同意管理与元素事件触发器
useScript
可组合函数使您完全控制脚本的加载方式和时机,可以通过提供自定义 trigger
或手动调用 load()
函数。
在此基础上,Nuxt Scripts 提供了高级触发器,以使其更容易。
const cookieConsentTrigger = useScriptTriggerConsent()
const { proxy } = useScript<{ greeting: () => void }>('/hello.js', {
// 仅在同意被接受后加载脚本
trigger: cookieConsentTrigger
})
// ...
function acceptCookies() {
cookieConsentTrigger.accept()
}
// greeting() 在用户接受 cookies 之前会被排队
proxy.greeting()
脚本打包
在许多情况下,我们从一个我们无法控制的域加载第三方脚本。这可能导致许多问题:
- 隐私:第三方脚本可以跨站点跟踪用户。
- 安全:第三方脚本可能被破坏并注入恶意代码。
- 性能:额外的 DNS 查询会减慢页面加载。
- 开发者体验:经过同意的脚本可能会被广告拦截器阻止。
为此,Nuxt Scripts 提供了一种将第三方脚本打包到您的公共目录而无需额外工作的方法。
useScript('https://cdn.jsdelivr.net/npm/js-confetti@latest/dist/js-confetti.browser.js', {
bundle: true,
})
现在,这个脚本将从 /_scripts/{hash}
以您的自己域名提供。
待续
正如我们所看到的,还有许多机会可以改进开发者和最终用户的第三方脚本。
Nuxt Scripts 的初始发布解决了 一些 这些问题,但我们还有很多工作要做。
下一步的规划项目包括:
我们欢迎您的贡献和支持。
开始使用
要开始使用 Nuxt Scripts,我们创建了一份 教程 来帮助您快速上手。
感谢
- Harlan Wilton - Nuxt (作者)
- Julien Huang - Nuxt (贡献者)
- Daniel Roe - Nuxt (贡献者)
- Chrome Aurora - Google (贡献者)
并感谢早期的贡献者们。