Skip to content

每次给博客搞拆迁,最让人割舍不下的,往往不是瓦片和墙皮,而是那些被我用到形成肌肉记忆的家具。由于Jekyll 和 Astro 说着不同方言,导致家具根本拿不到“异地安置指标”,我只能含泪签字

Photosuite 正是在这样的背景下诞生的

它由 Vite + Typescript 开发,拥有我博客图像的核心能力,包括:

  • 灯箱

  • EXIF 展示

  • 图片说明

  • 图片路径自动补全

Example Picture
ILCE-7CM2 · E 28-200mm F2.8-5.6 A071 · 51.0 mm · ƒ/3.5 · 1/1250 · ISO 1000 · 2025/10/4

上面的图片及其所有样式,仅通过下面这一行最普通的 Markdown 语法生成,而且我只需要输入文件名:

![Example Picture](example-picture.jpg)

设计思路

Photosuite 利用 Remark 和 Rehype 插件生态,在 Markdown 编译阶段完成图片处理,避免在运行时增加负担:

- 构建期
├→ Remark:补全图片 URL
└→ Rehype:读取图片 EXIF 并写入 HTML
- 运行期
├→ photosuite(opts) 入口:按需动态 import
├→ glightbox 模块:把图变成可点击灯箱
├→ imageAlts 模块:用 alt 自动生成 caption
└→ exif 模块:加载 EXIF 样式,清理空条

图片路径解析

如前所述,Photosuite 的首要职责是补全图片 URL

设计目标很简单:在 Markdown 中只写文件名,其余交给 Photosuite 处理

整体思路如下:

- 使用标准 Markdown 语法插入图片,只填写文件名
例:![Example Picture](demo.jpg)
- 配置一个基础 URL 作为图片域名
https://cdn.example.com/images
- 子目录来源有两种策略:
a. 通过 Frontmatter 指定图片目录(默认)
imageDir: 2025-12-22-photosuite
b. 以当前 Markdown 文件名作为目录(去除后缀)
2025-12-22-photosuite.md
- 最终生成的完整路径一致,例如:
https://cdn.example.com/images/2025-12-22-photosuite/demo.jpg
- 稍作调整即可实现按年 / 月分类等更复杂的目录结构。

实现逻辑:

  1. 使用 Remark 插件遍历 Markdown AST 中的所有 image 节点
  2. 判断图片 URL 是否为“短链接”(无协议、非绝对路径、非显式相对路径 ../
  3. 按配置策略拼接完整 URL(域名 + 目录 + 文件名)
  4. 重写 AST 节点的 url 属性

EXIF 展示

EXIF 本身并不是 Photosuite 的核心创新点,毕竟,也不是我实现的

我只是在 HTML 生成之前,我通过 exiftool-vendored.js 提取图片参数,并将其以文本形式注入到 DOM 中,仅此而已,零 JS 成本、零性能开销

遍历 HTML AST → 找到 img → 解析图片 → 提取 EXIF → 重写节点结构

HTTP 图片的临时下载策略

function isHttpUrl(u: string): boolean {
const x = new URL(u);
return x.protocol === "http:" || x.protocol === "https:";
}

对于网络图片,Photosuite 不会直接让 exiftool 处理 URL,而是

  • 下载到 os.tmpdir()
  • 生成随机文件名
  • finally 中清理
const dl = await downloadToTemp(src);
filePath = dl.path;
cleanup = dl.cleanup;

可配置字段

exiftool-vendored.js 解析的 EXIF 数据非常多。这里 Photosuite 做了封装处理,除默认字段外,还可以任意搭配

默认展示字段:

['Model', 'LensModel', 'FocalLength', 'FNumber', 'ExposureTime', 'ISO', 'DateTimeOriginal']

通过 formatField 做语义化输出:

case 'FNumber':
return `ƒ/${Number(value).toFixed(1)}`;
case 'ExposureTime':
return value >= 1 ? `${value}s` : `1/${Math.round(1 / value)}s`;
case 'ISO':
return `ISO ${value}`;

最终输出示例:

ILCE-7CM2 · E 28-200mm F2.8-5.6 A071 · 51.0 mm · ƒ/3.5 · 1/1250 · ISO 1000 · 2025/10/4

实际上,我最初我选择的是 MikeKovarik/exifr

它号称是速度最快、功能最全的 JavaScript EXIF 解析库
结果,我 Nikon z30 拍摄的照片它居然识别不出来机身?我尼康就低人一等吗!果断 PASS!

按需加载

Photosuite 的所有功能都是模块化设计的,样式同样如此

当你关闭某个功能时:

  • 对应的 JavaScript 不会加载

  • 相关 CSS 也不会出现在页面中

实现逻辑:

  1. 检查配置中的 scope(作用域选择器)是否存在于页面中

    • 若不存在,直接终止执行
  2. 根据功能开关配置:

    • enableLightbox, enableAlts, enableExif 并行、动态 import() 对应模块

DOM 标准化

Photosuite 设计了一套统一的 DOM 规范,供所有模块共享

核心思想是:

先把来源复杂的图片元素统一成稳定结构,再在此基础上扩展功能

开关 Photosuite 任意功能都会生成不同的结构,以下是所有功能开启时的最终结构:
<div class="photosuite-item">
<div class="photosuite-exif"></div>
<a class="glightbox" href="...">
<img ... />
</a>
<div class="photosuite-caption">alt</div>
</div>

关键逻辑

export function ensurePhotosuiteContainer(el: Element): HTMLElement {
let target: Element = el;
// 如果 img 被 a.glightbox 包裹,则提升包裹层级
if (
el.tagName.toLowerCase() === "img" &&
el.parentElement?.tagName.toLowerCase() === "a" &&
el.parentElement.classList.contains("glightbox")
) {
target = el.parentElement;
}
// 如果已经在 photosuite-item 中,直接复用
const parent = target.parentElement as HTMLElement | null;
if (parent?.classList.contains("photosuite-item")) {
return parent;
}
// 创建统一容器
const wrapper = document.createElement("div");
wrapper.className = "photosuite-item";
// 用 wrapper 替换 target
parent?.replaceChild(wrapper, target);
wrapper.appendChild(target);
return wrapper;
}

有了这个保证之后,后续逻辑可以完全不关心图片来源

// 查找主体
container.querySelector("img");
// 添加 UI(caption)
container.appendChild(caption);
// 查询数据:有就显示,没有就清理(EXIF)
container.querySelector(".photosuite-exif");

安装

Photosuite 已发布至 npm,可直接安装:

pnpm add photosuite
# or
npm install photosuite
# or
yarn add photosuite

快速开始

配置 Photosuite 非常简单,以Astro为例:

astro.config.js
import { defineConfig } from 'astro/config';
import photosuite from 'photosuite';
import "photosuite/dist/photosuite.css";
export default defineConfig({
integrations: [
photosuite({
// [必填] 生效范围选择器
// 建议限定在文章容器内,避免影响站点其他区域。支持多个选择器,用逗号分隔
scope: '#main',
})
]
});
PostDetails.astro
import "photosuite/dist/photosuite.css";

Photosuite 基于 Vite + TypeScript 开发,理论上适用于多种架构

如果您在配置中遇到任何问题,欢迎联系我,为爱发电,无偿奉献

后续计划

  1. 引入 Lozad.js 实现图片懒加载
  2. 构建期获取图片尺寸,生成占位符,避免布局抖动
  3. 适配更多博客架构
  4. 进一步优化 Photosuite 的按需加载机制

相关资料

如果 Photosuite 对您有帮助,欢迎在 GitHub 点个 ⭐️
它不会让代码跑得更快,但会让我写得更勤快

PS:如果您正在使用 Photosuite
欢迎告诉我您的博客地址,我会把它展示在项目页面中