zcbot/skills/ppt/templates/charts/CHART_STYLE_GUIDE.md

29 KiB
Raw Blame History

Chart SVG Style Guide

本文档定义了 templates/charts/ 下所有 SVG 图表模板的视觉规范。
新增或修改图表时 必须 遵循以下标准,确保全库视觉一致性。

0. 上游规范引用

本文档是 图表模板专用 的美学与实现规范。所有图表同时必须遵守项目级通用技术约束:

references/shared-standards.md — SVG 禁用特性黑名单、PPT 兼容性替代、Canvas 格式、tspan 内联规则、分组规范、阴影/叠加技术、后处理管线

以下章节摘录了 shared-standards 中与图表模板最密切相关的条目。完整细节(如 marker 条件约束、clipPath 条件约束、弧线路径计算公式等)请查阅上游文档。


1. 色彩系统 (Tailwind CSS Palette)

1.1 文本颜色

用途 色值 Tailwind Token 示例
主标题 #0F172A Slate 900 图表大标题
数值标签 #0F172A Slate 900 柱顶数值、关键指标
副标题 #64748B Slate 500 日期、单位说明
坐标轴标签 #64748B Slate 500 X/Y 轴刻度值
轴标题 / 图例 #475569 Slate 600 "年薪(万元)"、图例文字
数据来源 #94A3B8 Slate 400 页面底部来源说明
脚注 / 淡化提示 #CBD5E1 Slate 300 "各阶段可灵活调整"

1.2 主题色(数据系列)

色名 主色 深色(渐变终点) 用途
Blue #3B82F6 #2563EB 第 1 系列(默认首选)
Emerald #10B981 #059669 第 2 系列
Amber #F59E0B #D97706 第 3 系列
Violet #8B5CF6 #7C3AED 第 4 系列
Rose #FB7185 #E11D48 第 5 系列 / 警告
Pink #EC4899 #BE185D 对比组(如蝴蝶图女性)

径向渐变(如气泡图)使用亮色变体:#60A5FA#34D399#FBBF24#A78BFA#FB7185

1.3 语义色

用途 色值 说明
达标 / 正面 #10B981 Emerald 500
警告 / 中性 #F59E0B Amber 500
未达标 / 负面 #EF4444 Red 500
异常值标注 #F43F5E Rose 500

1.4 UI 辅助色

用途 色值 说明
坐标轴线 #94A3B8 Slate 400, stroke-width="2"
网格线 #E2E8F0#E0E0E0 stroke-dasharray="4,4"
中心分隔线 #CBD5E1 如象限十字线
卡片背景 #F8FAFC / #F8F9FA Slate 50
卡片描边 #E2E8F0 Slate 200
行分隔线 #F1F5F9 Slate 100极淡
Tint 背景(蓝) #EFF6FF Blue 50
Tint 背景(绿) #ECFDF5 Emerald 50
Tint 背景(红) #FFF1F2 Rose 50
Tint 背景(黄) #FFFBEB Amber 50

2. 排版规范

2.1 字体栈

font-family="-apple-system, BlinkMacSystemFont, 'Segoe UI', 'PingFang SC', 'Microsoft YaHei', sans-serif"
  • 纯英文场景可省略 'PingFang SC', 'Microsoft YaHei'
  • 禁止 使用 @font-face、外部字体、<style> 标签

2.2 字号层级

层级 字号 font-weight 用途
H1 34px bold (700) 图表主标题
H2 22px 600 区域标题(如"详细数据"
Body L 18-20px 600 关键数值、百分比
Body M 15-16px 600 数据标签、分类名
Body S 14px 正常 副标题、图例、来源
Caption 12-13px 正常 坐标轴刻度、注释

最小字号下限12px。所有文本不得小于 12px。

2.3 tspan 规范

所有 <text> 元素的文本内容 必须 包裹在 <tspan> 中:

<!-- 正确 -->
<text x="60" y="80" font-size="34" fill="#0F172A">
    <tspan>图表标题</tspan>
</text>

<!-- 错误 -->
<text x="60" y="80" font-size="34" fill="#0F172A">图表标题</text>

2.4 内联格式化规则shared-standards SS4

单逻辑行 = 单 <text>。同一行内需要多色/多粗细时,用内联 <tspan> 实现,不要用多个并排 <text>

<!-- 正确:一个 text frame三个 run -->
<text x="100" y="200" font-size="24" fill="#333333">
  实现<tspan fill="#3B82F6" font-weight="bold">10倍</tspan>效率提升
</text>

<!-- 错误:三个独立 text framePPT 中无法作为一行编辑 -->
<text x="100" y="200">实现</text>
<text x="160" y="200" fill="#3B82F6">10倍</text>
<text x="240" y="200">效率提升</text>

内联 tspan 不得 携带 x / y / dy,否则后处理会将其拆分为独立 text frame。dx 可用于微调字距。

2.5 数据高亮默认行为

图表中的关键数据文本应默认高亮:

  • 数值结果 — 百分比、倍数、金额 → <tspan fill="主题色" font-weight="bold">
  • 对比项 — 增/减、达标/未达标 → 语义色(绿/红)
  • 不高亮 — 连接词、普通动词、结构性文字(轴标签、图例、页码)

3. 阴影滤镜

<filter> 本身是允许的、且是 PPT 阴影/发光的官方推荐路径(详见本节末尾的"禁用列表"说明)。本节统一阴影 primitive 写法——使用 feFlood 方案,禁止 <filter> 内部使用 <feComponentTransfer>

<filter id="chartShadow" x="-15%" y="-15%" width="130%" height="130%">
    <feGaussianBlur in="SourceAlpha" stdDeviation="2-4"/>
    <feOffset dx="0" dy="1-3" result="offsetBlur"/>
    <feFlood flood-color="#0F172A" flood-opacity="0.08-0.15" result="shadowColor"/>
    <feComposite in="shadowColor" in2="offsetBlur" operator="in" result="shadow"/>
    <feMerge>
        <feMergeNode in="shadow"/>
        <feMergeNode in="SourceGraphic"/>
    </feMerge>
</filter>

参数参考

场景 stdDeviation dy flood-opacity
重型元素(箭头、卡片) 4-6 2-4 0.12-0.15
中型元素(柱子、箱体) 2-3 1-2 0.10-0.15
轻型元素(底部卡片) 4-6 2-4 0.06-0.08

禁用列表

  • flood-color="#000000" → 必须用 #0F172A
  • <feComponentTransfer> + <feFuncA slope=...> → 用 <feFlood flood-color flood-opacity> 替代
  • flood-opacity > 0.20 → 阴影过重,最大 0.15-0.20

被禁的是 sub-element不是 <filter> 本身。 <filter> 是 PPT Master 允许的、官方推荐的阴影/发光路径(见 shared-standards.md §1 黑名单不含 filter、§6 把 filter shadow 列为 drop-shadow 的官方实现),转换器 svg_to_pptx/drawingml_styles.py 也主动把 feGaussianBlur + feOffset + feFlood + feComposite + feMerge(以及 feDropShadow 简写)映射成 DrawingML <a:outerShdw>

单独禁 feComponentTransfer/feFuncA(slope) 的原因:它物理上只能调透明度、无法携带颜色。转换器读到 feFuncA slope 时只把它当作 alpha颜色字段保持默认 '000000'——SVG 端看起来阴影颜色正常(因为 SourceAlpha 本身是黑),但导出到 PPTX 后阴影颜色会被定死成纯黑 #000000,与同页其他用 feFlood flood-color="#0F172A" 的卡片产生肉眼可见的冷暖色差。

简言之:用 filter 没问题,但 primitive 必须能把"颜色"显式表达出来;只能表达"透明度"的 primitive 是被禁的。

阴影使用原则shared-standards SS6

阴影是美学成分,不是默认处理。 克制而非丰富才能产生"经过设计"的感觉。 "阴影被感知而非被看见" 是高端美学标准。

应加阴影:浮在照片/彩色面板上方的卡片、唯一的主 CTA、叠加层tooltip、callout

不应加阴影:背景面板/分隔条、网格中平等的同级卡片、已有描边/渐变的容器、正文段落容器、装饰线/图标、深色背景上(黑色阴影不可见)

每页预算:最多 2-3 个带阴影元素。第 4 个需要阴影时,先移除现有某个的阴影。

统一光源:同页所有 feOffsetdx/dy 方向必须一致(默认 dx=0, dy=正值,光从上方来)。

两级高度上限

层级 场景 dy stdDeviation flood-opacity
地面(无阴影) 背景、同级网格卡片、分隔线、正文容器
静止 照片/面板上的卡片、次级 callout 2-4 4-8 0.06-0.10
抬升 主 CTA、焦点/推荐卡片、覆盖层 6-10 10-16 0.12-0.20

不要堆叠:阴影 + 描边 + 圆角 + 渐变填充同时出现 = 模板感。容器的"看我"预算很小,选其一即可。


4. 渐变规范

4.1 线性渐变(柱状/条形图)

<linearGradient id="barGrad1" x1="0%" y1="0%" x2="0%" y2="100%">
    <stop offset="0%" style="stop-color:#3B82F6;stop-opacity:1" />
    <stop offset="100%" style="stop-color:#2563EB;stop-opacity:1" />
</linearGradient>
  • 方向:从亮到深(顶到底 或 左到右)
  • 每个渐变 ID 应语义化:barGrad1leftGradactualBarBlue

4.2 径向渐变(气泡图)

<radialGradient id="bubbleGrad1" cx="30%" cy="30%">
    <stop offset="0%" style="stop-color:#60A5FA;stop-opacity:0.9" />
    <stop offset="100%" style="stop-color:#2563EB;stop-opacity:0.7" />
</radialGradient>
  • 高光偏左上方 (cx="30%" cy="30%")
  • 边缘 opacity 降低至 0.7,制造通透感

5. 结构规范

5.1 层级分组shared-standards SS4 Grouping

使用 <g id="..."> 进行语义分组,便于 PPT 中逐个操作/动画:

<g id="chartArea">        <!-- 图表主体 -->
    <g id="bar-1">...</g>  <!-- 每个数据元素独立分组 -->
    <g id="bar-2">...</g>
</g>
<g id="legend">            <!-- 图例区域 -->
    <g id="legend-high">...</g>
</g>
<g id="detailList">        <!-- 详情面板 -->
    <g id="list-items">
        <g id="item-1">...</g>
    </g>
</g>

分组单元参考(来自 shared-standards

分组单元 包含内容
卡片/面板 背景 rect + 阴影(如适用)+ 图标 + 标题 + 正文
流程步骤 编号圆 + 图标 + 标签 + 描述
列表项 圆点/编号 + 图标 + 标题 + 描述
图标-文字组合 图标元素 + 相邻标签
页头 标题 + 副标题 + 装饰
装饰集群 相关装饰形状(环、球、点)

命名约定:使用描述性 id(如 card-1step-discoverheaderfooter)。

只有 <g opacity="..."> 被禁止(见 SS2。纯结构 <g> 是必需的。

5.2 viewBox

固定为 0 0 1280 720PPT 16:9不可修改。

5.3 背景

首行始终为白色全屏背景:

<rect width="1280" height="720" fill="#FFFFFF"/>

5.4 数据来源

位于页面底部,固定格式:

<text x="60" y="695" font-family="..." font-size="14" fill="#94A3B8">
    <tspan>数据来源: XXX</tspan>
</text>

6. SVG 禁用特性与兼容性shared-standards SS1-2

6.1 绝对禁止

禁用特性 替代方案
HTML 命名实体(&nbsp; &mdash; &copy; &ndash; &reg; &hellip; &bull; …) 直接写原生 Unicode 字符( © ® NBSP …)
文本/属性值中裸写 & < > " ' 必须写成 XML 实体 &amp; &lt; &gt; &quot; &apos;
<style> / class 内联属性(id<defs> 内合法)
<foreignObject> <text> + <tspan>
mask 叠加遮罩矩形 / gradient overlay
<symbol> + <use> 直接写出完整元素
textPath 手动排列 <text>
@font-face 系统字体栈
<animate*> / <set> PPT 侧处理动画)
<script> / event 属性
<iframe>

6.2 PPT 兼容性替代

禁止语法 正确替代
fill="rgba(255,255,255,0.1)" fill="#FFFFFF" fill-opacity="0.1"
<g opacity="0.2">...</g> 在每个子元素上单独设置 fill-opacity / stroke-opacity
<image opacity="0.3"/> 在 image 后叠加 <rect fill="背景色" opacity="0.7"/>

6.3 条件允许

特性 条件 转换结果
marker-start / marker-end <marker><defs> 中,orient="auto",形状为三角/菱形/圆 DrawingML <a:headEnd> / <a:tailEnd>
clipPath on <image> <clipPath><defs> 中,单子元素,仅用于 image DrawingML <a:prstGeom> / <a:custGeom>
stroke-dasharray 使用预设值 4,4 / 2,2 / 8,4 / 8,4,2,4 PPTX <a:prstDash>
text-decoration underline / line-through PPTX 原生文本格式
transform="rotate(...)" 所有元素类型均支持 PPTX <a:xfrm rot="...">

完整条件约束见 shared-standards.md SS1.1marker 约束)和 SS1.2clipPath 约束)。

6.4 虚线预设对照

SVG 值 PPTX 预设 适用场景
4,4 Dash 通用虚线、分隔线
2,2 Dot (sysDot) 占位轮廓、细边框
8,4 Long dash 时间线连接、流程箭头
8,4,2,4 Long dash-dot 技术图纸、尺寸线

7. 旧色映射速查表

在维护旧模板时,使用以下映射快速替换:

旧色 (Material/Flat) 新色 (Tailwind) 角色
#2C3E50 #0F172A 主文本
#7F8C8D #64748B 副文本
#5D6D7E #475569 图例文本
#95A5A6 #94A3B8 数据来源
#BDC3C7 #CBD5E1 淡化元素
#2196F3 / #1976D2 #3B82F6 / #2563EB 蓝色系列
#4CAF50 / #388E3C #10B981 / #059669 绿色系列
#FF9800 / #F57C00 #F59E0B / #D97706 橙色系列
#E91E63 #F43F5E 异常值
#000000 (shadow) #0F172A 阴影底色

8. 占位内容规范 (Placeholder Content Strategy)

既然这些 SVG 文件是供 AI 后续调用的“模板”,它们的核心价值在于展示 图形结构、排版约束与视觉空间,而不是传递真实的业务数据。因此,写入模板的文本内容应遵循以下“占位原则”:

8.0 全英文原则 (English-Only Rule)

强制要求:所有图表模板中的占位文本(包括标题、副标题、坐标轴、图例、数据节点、详情描述及底部来源说明)必须全部使用英文编写

  • 目的:确保后续自动化管线中的 LLM 能够更精准地进行语义理解和结构化内容映射,同时英文单词的天然长度特征更易于在模板中展示排版时的换行逻辑与空间边界。

8.1 结构边界演示

  • 展示最大宽度/换行逻辑:刻意使用典型长度的字符串(如两到三个词的短语、多行 tspan)来明确展示文本框的边界。这样能确保 AI 填入真实文本时有直观的参考,防止溢出。
  • 展示数据格式:使用能体现完整格式特征的占位数值(如 $1,234.5M98.5%)而不仅是简单的 10,以验证符号和字符宽度是否适配。

8.2 通用性与中立性

  • 使用通用、专业的商业占位符,避免过于垂直或具象的特定业务数据(除非该模板本身具有强烈的行业属性)。
  • 推荐做法:使用 Category AQ1 RevenueStrategic ObjectivePhase 01
  • 避免做法使用具体的长篇现实数据如“某某品牌2023年特种设备销量分析”

8.3 视觉平衡

  • 占位文本应当在视觉上保持图表的平衡性(例如蝴蝶图左右文本长度应大致相等,列表文本应长短错落有致),以便让人一眼看清图表的布局设计意图。

9. 注册到 charts_index.json

新增 SVG 模板后,必须charts_index.json 中登记,否则 Strategist 选型时不会发现它。

9.1 字段规范

"<key>": {
  "summary": "Pick for <内容形态 + 规模>. Skip if <反例 → 替代模板>."
}
  • key = SVG 文件名去掉 .svg,下划线小写(如 bullet_chart
  • summary选型句,不是描述句。语法见 meta.summaryGrammar:先说什么时候选它,再用 Skip if ... (use <other_key>) 指向最容易混淆的兄弟模板
  • meta.total 同步 +1

不需要 label / categories / quickLookup / keywords —— 这些都已经移除。Strategist 全量读取 summary 列表后语义匹配,不依赖任何预计算索引。注意summary 是英文,但 source 文档常含中文/行业术语("中台"、"架构图"、"管道"Strategist 自己负责语义翻译再匹配。如果一个模板的命中强依赖某个中文短语,把它的英文等价物写进 summary 的 Pick 子句里。

9.2 反例

只写"是什么""summary": "Bidirectional comparison chart for two datasets" 写"何时选""summary": "Pick for two mirrored datasets sharing a common axis (age pyramid, A/B). Skip for >2 sides (use grouped_bar_chart)."

summary 过长(>400 字符)—— 选型时反而难抓重点,目标在 150-300 字符。

Why not stricter:单一结构模板常需覆盖多个商业框架/场景(如 quadrant_text_bullets 覆盖 SWOT + Ansofftop_down_tree 覆盖 org + OKRsummary 需要列出关键词锚点("principles, key takeaways, action items" 这种)才能让 Strategist 语义命中"非数字结构页",所以 100-180 字符的旧基线在结构-派命名后已经太紧。


10. 检查清单

新增或修改图表后,逐项检查:

基础校验

  • xmllint --noout 通过
  • viewBox 为 0 0 1280 720
  • 首行为白色背景 <rect width="1280" height="720" fill="#FFFFFF"/>

色彩

  • 无旧色残留(grep 验证,见下方命令)
  • 阴影 flood-color#0F172Aopacity 小于等于 0.20
  • 数据来源用 #94A3B8

排版

  • font-size < 12 的文本
  • 所有 <text> 内容包裹 <tspan>
  • 同一行多格式用内联 <tspan>多个并排 <text>
  • 内联 <tspan> 不携带 x / y / dy
  • 标题 34px、副标题 18px、来源 14px

结构

  • 主要元素有语义化 <g id="...">
  • <style>class<foreignObject>maskrgba()
  • <g> 标签无 opacity 属性
  • 文本字符为原生 Unicode © NBSP 等),无 HTML 命名实体(&nbsp; &mdash; &copy; 等);裸 & < > 已转义为 &amp; &lt; &gt;

阴影

  • 使用 feFlood 方案(非 feComponentTransfer
  • 同页阴影 dx/dy 方向一致
  • 每页带阴影元素不超过 3 个

注册(仅新增模板时)

  • charts_index.jsoncharts.<key> 已登记 summary 字段
  • summary 写成选型句(Pick for ... Skip if ... (use <other>)),不是描述句
  • summary 长度控制在 150-300 字符(>400 字符要重写);如果模板覆盖多个商业框架/场景,可放宽到 350 字符以塞下关键词锚点
  • meta.total 同步 +1

坐标校准标记calculator-supported 图表必填)

  • 矩形坐标系图表bar / horizontal_bar / grouped_bar / stacked_bar / line / area / stacked_area / scatter / waterfall / pareto / butterfly包含 <!-- chart-plot-area: x_min,y_min,x_max,y_max --> 标记
  • Pie / donut / radar 图表包含 <!-- chart-plot-area: <type> | center: cx,cy | radius: r --> 标记
  • 标记位于 <g id="chartArea"> 内、坐标轴之后、数据元素之前
  • 坐标值与轴线的实际 SVG 坐标一致

验证命令

# 一键校验
f="your_chart.svg"
xmllint --noout "skills/ppt/templates/charts/$f" && echo "XML OK" || echo "XML FAIL"
echo "Old colors:" && grep -c '#2C3E50\|#7F8C8D\|#95A5A6\|#5D6D7E\|#000000' "skills/ppt/templates/charts/$f"
echo "Small fonts:" && grep -c 'font-size="[0-9]"' "skills/ppt/templates/charts/$f"

11. 卡片容器图式 (Card Container Patterns)

容器卡是 PPT Master 中复用率最高的视觉单元KPI 卡、分区卡、信息卡)。下面三种图式是经过验证、与 PPTX 往返兼容的"参考实现",新增模板优先沿用,不要发明等价但实现脏的替代。

11.1 半圆角分区头 (Half-Rounded Section Tab)

用途:给卡片或区块加一个有色"标签头"标识分类S/W/O/T、Political/Economic、自我介绍/获奖等)。比纯文字大标题更易识别,比独立标签条更紧凑。

两种形态——根据 tab 的"视觉锚点"在上还是在下选择:

形态 形状 视觉语义 典型场景
上圆下方 (圆顶角) 顶部两角圆,底部两角直 从卡片"长出来"的标签 分区卡头部、quadrant 标题、信息卡分类
上方下圆 (圆底角) 顶部两角直,底部两角圆 从页头/章节条"悬挂下来"的吊牌 章节锚点、页头分隔条延伸、目录跳转标记

两种形态的共同要求:只圆一对角,整条 path 直接画出来。不要用"全圆角矩形 + 同色矩形盖底/盖顶"的 hack往返到 PPTX 时会变成两个独立对象,编辑时颜色容易脱钩)。

实现一:上圆下方(默认)

<!-- 模板:宽 W、高 H、圆角 R左上原点 (x, y) -->
<path d="M {x+R} {y} h {W-2R} a {R} {R} 0 0 1 {R} {R} v {H-R} h -{W} v -{H-R} a {R} {R} 0 0 1 {R} -{R} Z"
      fill="#2563EB"/>

<!-- 实例240×50, r=25, 起点 (245, 140) -->
<path d="M 245 140 h 190 a 25 25 0 0 1 25 25 v 25 h -240 v -25 a 25 25 0 0 1 25 -25 Z" fill="#2563EB"/>

实现二:上方下圆(悬挂吊牌)

<!-- 模板:宽 W、高 H、圆角 R左上原点 (x, y) -->
<path d="M {x} {y} h {W} v {H-R} a {R} {R} 0 0 1 -{R} {R} h -{W-2R} a {R} {R} 0 0 1 -{R} -{R} Z"
      fill="#2563EB"/>

<!-- 实例240×50, r=25, 起点 (245, 140) -->
<path d="M 245 140 h 240 v 25 a 25 25 0 0 1 -25 25 h -190 a 25 25 0 0 1 -25 -25 Z" fill="#2563EB"/>

禁用反例PEST/SWOT/comparison_columns 旧实现中常见):

<!-- ❌ 不要这样写:用全圆角矩形 + 白色矩形覆盖一边圆角 -->
<rect width="260" height="120" rx="12" fill="#EFF6FF"/>
<rect y="100" width="260" height="20" fill="#EFF6FF"/>

底部覆盖矩形在 SVG→PPTX 往返时会变成一个独立的、跟头部颜色绑定的矩形对象PPT 里编辑头部颜色时容易漏改、视觉会"穿帮"。

11.2 嵌套卡片描边 (Nested Card Border)

用途:让卡片有"被描边"的层次感,但避免 stroke。stroke 在 PPTX 中常被渲染为细线分层,且与阴影叠加易产生模板感。

做法:外层浅灰圆角 rect + 内层白色稍小圆角 rect两层之间留出 820px 缝即可形成"边框"效果。

<!-- 外层"边框"层 -->
<rect x="60" y="140" width="560" height="255" rx="20" fill="#F1F5F9"/>
<!-- 内层白色内容卡(内缩 20px半径变小 -->
<rect x="80" y="210" width="520" height="165" rx="12" fill="#FFFFFF"/>

适用条件

  • 当卡片上方还有 §11.1 的分区头时,外层框充当头部的"背板"
  • 同页只用 一种 描边表达:外层框 OR stroke OR 阴影,不要同时用(参见 §3 阴影使用原则)

11.3 卡片网格作为内容页骨架 (Card Grid as Page Skeleton)

用途:当一页要并列展示 4 个平等的方面pillar / aspect / quadrant优先用 2×2 网格而非垂直叠加。

网格尺寸建议1280×720 画布):

网格 单卡宽 × 间距 起始 (x, y)
2×2 560 × 255 40 (60, 140) (660, 140) (60, 420) (660, 420)
2×3 (横) 370 × 260 25 (50, 130) 行距 290
1×3 (横长) 400 × 540 30 (60, 130) 列距 430
1×4 (顶) 280 × 250 20 (60, 150) 列距 300

判定"4 个并列方面" → 2×2"3 个并列方面" → 1×3"6 个能力点" → 2×3"4 个关键指标" → 1×4。page_rhythmbreathing 的页面 不要 用卡片网格(见 executor-base.md §2.1)。

11.5 倾斜虚线连接箭头 (Diagonal Dashed Connector)

用途:表达"跨象限/跨层级"的关系——优先级迁移、影响传导、虚线汇报、对角趋势。水平/垂直箭头表达的是"流程进度",倾斜虚线箭头表达的是"关系或方向引导",两者语义不一样。

做法:单条 <line> + stroke-dasharray="6 5" + marker-end。需要为这条线单独定义一个 marker不复用主流程图的箭头颜色建议用 Slate 600 #475569 表达"建议性、非强制"的色调)。

<defs>
  <marker id="migrationArrow" markerWidth="12" markerHeight="12"
          refX="10" refY="6" orient="auto" markerUnits="strokeWidth">
    <path d="M 0,0 L 10,6 L 0,12 Z" fill="#475569"/>
  </marker>
</defs>

<!-- 从 Q4 (右下) 指向 Q2 (左上) 的优先级迁移箭头 -->
<line x1="850" y1="605" x2="385" y2="200"
      stroke="#475569" stroke-width="2"
      stroke-dasharray="6 5" stroke-linecap="round"
      marker-end="url(#migrationArrow)"/>

<!-- 中段标签:白底胶囊压在箭头上,避免视觉打架 -->
<rect x="525" y="385" width="190" height="28" rx="14"
      fill="#FFFFFF" stroke="#CBD5E1" stroke-width="1"/>
<text x="620" y="403" text-anchor="middle" font-size="12"
      font-weight="700" fill="#475569" letter-spacing="1">PRIORITY MIGRATION</text>

配对要求:每条倾斜虚线箭头必须配一个中段标签(小胶囊或一行文字),否则读者会困惑"这条线在说什么"。无标签的箭头只允许出现在水平/垂直流程中(如 process_flow)。

11.6 接地椭圆 (Ground Anchor Ellipse) — 非 filter 的深度表达

用途:让"漂浮在卡片上的圆形/icon/人物头像/奖杯/角色徽章"获得"接触地面"的视觉锚定,但不使用 <filter> 阴影

为什么有用

  1. PPTX 原生圆/椭圆对象,跨渲染器一致,不会被解析为 <a:outerShdw>(避免阴影颜色丢失或重排问题)
  2. 跟 §3 「克制阴影」呼应——一页阴影预算上限 2-3 个,剩下需要"深度"的元素可以走这条路
  3. 比 filter 阴影更容易在 PPT 中二次编辑(用户可以直接拖、改色、删除)

做法:在浮动元素正下方画一个横扁椭圆ry << rx),低透明度,颜色用主体色或 Slate 900

<!-- 头像/徽章下方的接地阴影板cy 比头像底边低 10-15px -->
<ellipse cx="80" cy="172" rx="70" ry="5" fill="#0F172A" opacity="0.10"/>
<!-- 然后再画头像本体(顺序很重要,椭圆必须先画) -->
<circle cx="80" cy="80" r="80" fill="#E2E8F0"/>

参数参考

浮动元素半径 椭圆 rx 椭圆 ry opacity
30-50 px r × 0.85 3-4 0.10-0.15
50-100 px r × 0.85 5-6 0.10-0.12
100+ px r × 0.85 7-9 0.08-0.10

颜色:默认 #0F172A(中性深灰),可改为主体色的深色变体(如人物头像下用 #1E3A8A)表达"品牌色阴影"。

禁用:不要把椭圆画成正圆或近正圆(ry/rx > 0.25 就显得失真)。也不要叠在<filter> 阴影上——挑一种就够。

11.7 双向交互箭头 (Bidirectional Interaction Arrows)

用途:表达"请求/响应"、"推/拉"、"上行/下行"、"供给/需求"等成对关系。区别于单向流程箭头。

做法:两条平行的 <line> + 不同颜色的 marker-end,方向相反,每条线都必须带动作标签

<defs>
  <marker id="reqArrow" markerWidth="10" markerHeight="10" refX="9" refY="5"
          orient="auto" markerUnits="strokeWidth">
    <path d="M0,0 L10,5 L0,10 Z" fill="#3B82F6"/>
  </marker>
  <marker id="respArrow" markerWidth="10" markerHeight="10" refX="9" refY="5"
          orient="auto" markerUnits="strokeWidth">
    <path d="M0,0 L10,5 L0,10 Z" fill="#10B981"/>
  </marker>
</defs>

<!-- 请求:左到右,蓝色 -->
<line x1="380" y1="250" x2="926" y2="250" stroke="#3B82F6" stroke-width="2.5"
      marker-end="url(#reqArrow)"/>
<rect x="500" y="216" width="280" height="26" rx="11" fill="#FFFFFF"
      stroke="#3B82F6" stroke-width="1"/>
<text x="640" y="234" text-anchor="middle" font-size="14" font-weight="700"
      fill="#3B82F6">① Login Request · POST /auth/login</text>

<!-- 响应:右到左,绿色 -->
<line x1="926" y1="290" x2="384" y2="290" stroke="#10B981" stroke-width="2.5"
      marker-end="url(#reqArrow)"/>
<!-- ...同样配标签... -->

配色约定请求侧initiator用蓝色 #3B82F6、响应侧responder用绿色 #10B981。如果是对等关系(如 A↔B 协同),统一用 Slate 600 #475569 不区分颜色。

禁用:不允许画"裸线"——双向箭头每条都必须带标签说明动作;否则读者无法分辨方向语义。

11.8 参考实现

图式 参考模板
§11.1 半圆角分区头(上圆下方) quadrant_text_bullets.svg, labeled_card.svg, vertical_pillars.svg, comparison_columns.svg
§11.2 嵌套卡片描边 labeled_card.svg
§11.3 2×2 卡片网格 kpi_cards.svg, quadrant_text_bullets.svg, labeled_card.svg
§11.3 2×3 卡片网格 icon_grid.svg
§11.3 1×3/1×4 卡片网格 comparison_columns.svg, vertical_pillars.svg
§11.5 倾斜虚线连接箭头 matrix_2x2.svg
§11.6 接地椭圆 team_roster.svg
§11.7 双向交互箭头 client_server_flow.svg