Files
tyapi-server/internal/shared/pdf/font_manager.go

356 lines
10 KiB
Go
Raw Normal View History

2025-12-03 12:03:42 +08:00
package pdf
import (
"os"
2025-12-03 15:53:31 +08:00
"path/filepath"
2026-03-11 17:23:09 +08:00
"runtime"
2025-12-03 12:03:42 +08:00
"github.com/jung-kurt/gofpdf/v2"
"go.uber.org/zap"
)
// FontManager 字体管理器
type FontManager struct {
logger *zap.Logger
chineseFontName string
chineseFontLoaded bool
watermarkFontName string
watermarkFontLoaded bool
2026-03-16 12:32:41 +08:00
bodyFontName string
bodyFontLoaded bool
2025-12-03 12:03:42 +08:00
}
// NewFontManager 创建字体管理器
func NewFontManager(logger *zap.Logger) *FontManager {
2025-12-04 12:56:39 +08:00
return &FontManager{
2025-12-03 12:03:42 +08:00
logger: logger,
chineseFontName: "ChineseFont",
watermarkFontName: "WatermarkFont",
2026-03-16 12:32:41 +08:00
bodyFontName: "BodyFont",
2025-12-03 12:03:42 +08:00
}
}
2025-12-03 15:53:31 +08:00
// LoadChineseFont 加载中文字体到PDF
2025-12-03 12:03:42 +08:00
func (fm *FontManager) LoadChineseFont(pdf *gofpdf.Fpdf) bool {
if fm.chineseFontLoaded {
return true
}
2025-12-03 15:53:31 +08:00
fontPaths := fm.getChineseFontPaths()
2025-12-03 12:03:42 +08:00
if len(fontPaths) == 0 {
2025-12-04 12:30:33 +08:00
// 字体文件不存在,使用系统默认字体,不记录警告
2025-12-03 12:03:42 +08:00
return false
}
2025-12-03 15:53:31 +08:00
// 尝试加载字体
2025-12-03 12:03:42 +08:00
for _, fontPath := range fontPaths {
2025-12-03 15:53:31 +08:00
if fm.tryAddFont(pdf, fontPath, fm.chineseFontName) {
2025-12-03 12:03:42 +08:00
fm.chineseFontLoaded = true
return true
}
}
2025-12-04 12:30:33 +08:00
// 无法加载字体,使用系统默认字体,不记录警告
2025-12-03 12:03:42 +08:00
return false
}
2025-12-03 15:53:31 +08:00
// LoadWatermarkFont 加载水印字体到PDF
2025-12-03 12:03:42 +08:00
func (fm *FontManager) LoadWatermarkFont(pdf *gofpdf.Fpdf) bool {
if fm.watermarkFontLoaded {
return true
}
2025-12-03 15:53:31 +08:00
fontPaths := fm.getWatermarkFontPaths()
2025-12-03 12:03:42 +08:00
if len(fontPaths) == 0 {
return false
}
2025-12-03 15:53:31 +08:00
// 尝试加载字体
2025-12-03 12:03:42 +08:00
for _, fontPath := range fontPaths {
2025-12-03 15:53:31 +08:00
if fm.tryAddFont(pdf, fontPath, fm.watermarkFontName) {
2025-12-03 12:03:42 +08:00
fm.watermarkFontLoaded = true
return true
}
}
return false
}
2026-03-16 12:32:41 +08:00
// LoadBodyFont 加载正文用宋体(用于描述、详情、说明、表格文字等)
func (fm *FontManager) LoadBodyFont(pdf *gofpdf.Fpdf) bool {
if fm.bodyFontLoaded {
return true
}
fontPaths := fm.getBodyFontPaths()
for _, fontPath := range fontPaths {
if fm.tryAddFont(pdf, fontPath, fm.bodyFontName) {
fm.bodyFontLoaded = true
return true
}
}
return false
}
2025-12-03 15:53:31 +08:00
// tryAddFont 尝试添加字体(统一处理中文字体和水印字体)
func (fm *FontManager) tryAddFont(pdf *gofpdf.Fpdf, fontPath, fontName string) bool {
2025-12-03 12:03:42 +08:00
defer func() {
if r := recover(); r != nil {
2025-12-04 13:20:03 +08:00
fm.logger.Error("添加字体时发生panic",
zap.String("font_path", fontPath),
zap.String("font_name", fontName),
zap.Any("panic_value", r),
)
2025-12-03 12:03:42 +08:00
}
}()
2025-12-04 13:20:03 +08:00
// 确保路径是绝对路径
2025-12-04 10:35:11 +08:00
absFontPath, err := filepath.Abs(fontPath)
if err != nil {
2025-12-04 13:20:03 +08:00
fm.logger.Warn("无法获取字体文件绝对路径",
zap.String("font_path", fontPath),
zap.Error(err),
)
2025-12-04 10:35:11 +08:00
absFontPath = fontPath
}
2025-12-04 13:20:03 +08:00
fm.logger.Debug("尝试添加字体",
2025-12-04 13:42:32 +08:00
zap.String("font_path", absFontPath),
2025-12-04 13:20:03 +08:00
zap.String("font_name", fontName),
)
// 检查文件是否存在
fileInfo, err := os.Stat(absFontPath)
if err != nil {
fm.logger.Debug("字体文件不存在",
zap.String("font_path", absFontPath),
zap.Error(err),
)
return false
}
if !fileInfo.Mode().IsRegular() {
fm.logger.Warn("字体路径不是普通文件",
zap.String("font_path", absFontPath),
)
2025-12-04 10:35:11 +08:00
return false
}
2025-12-04 13:42:32 +08:00
// 确保路径是绝对路径gofpdf在Output时需要绝对路径
// 首先确保absFontPath是绝对路径
if !filepath.IsAbs(absFontPath) {
fm.logger.Warn("字体路径不是绝对路径,重新转换",
zap.String("original_path", absFontPath),
)
if newAbsPath, err := filepath.Abs(absFontPath); err == nil {
absFontPath = newAbsPath
}
}
2025-12-04 16:17:27 +08:00
2025-12-04 13:42:32 +08:00
// 使用filepath.ToSlash统一路径分隔符Linux下使用/
// 注意ToSlash不会改变路径的绝对/相对性质,只统一分隔符
2025-12-04 13:28:03 +08:00
normalizedPath := filepath.ToSlash(absFontPath)
2025-12-04 16:17:27 +08:00
2026-03-11 17:23:09 +08:00
// 在 Linux 下,绝对路径通常以 / 开头;在 Windows 下则可能以盘符 (C:/...) 开头。
// 这里只要保证 normalizedPath 非空即可,具体格式交给 gofpdf 处理,避免在 Windows 下误判。
if len(normalizedPath) == 0 {
fm.logger.Error("字体路径转换后为空无法添加到PDF",
2025-12-04 13:42:32 +08:00
zap.String("abs_font_path", absFontPath),
2026-03-11 17:23:09 +08:00
zap.String("font_name", fontName),
2025-12-04 13:42:32 +08:00
)
2026-03-11 17:23:09 +08:00
return false
2025-12-04 13:42:32 +08:00
}
2026-03-11 17:23:09 +08:00
// 额外记录当前平台,方便排查路径格式问题
fm.logger.Debug("字体路径平台信息",
zap.String("goos", runtime.GOOS),
zap.String("normalized_path", normalizedPath),
)
2025-12-04 13:42:32 +08:00
2025-12-04 13:28:03 +08:00
fm.logger.Debug("准备添加字体到gofpdf",
2025-12-04 13:42:32 +08:00
zap.String("original_path", fontPath),
2025-12-04 13:28:03 +08:00
zap.String("abs_path", absFontPath),
zap.String("normalized_path", normalizedPath),
zap.String("font_name", fontName),
)
2025-12-04 13:42:32 +08:00
// gofpdf v2使用AddUTF8Font添加支持UTF-8的字体
2025-12-04 14:21:58 +08:00
// 注意gofpdf在Output时可能会重新解析路径必须确保路径格式正确
// 记录传递给gofpdf的实际路径
fm.logger.Info("添加字体到gofpdf",
zap.String("font_path", normalizedPath),
zap.String("font_name", fontName),
zap.Bool("is_absolute", len(normalizedPath) > 0 && normalizedPath[0] == '/'),
)
2025-12-04 16:17:27 +08:00
2025-12-04 13:28:03 +08:00
pdf.AddUTF8Font(fontName, "", normalizedPath) // 常规样式
pdf.AddUTF8Font(fontName, "B", normalizedPath) // 粗体样式
2025-12-03 15:53:31 +08:00
// 验证字体是否可用
2025-12-04 13:42:32 +08:00
// 注意gofpdf可能在AddUTF8Font时不会立即加载字体而是在Output时才加载
// 所以这里验证可能失败,但不一定代表字体无法使用
2025-12-03 15:53:31 +08:00
pdf.SetFont(fontName, "", 12)
testWidth := pdf.GetStringWidth("测试")
if testWidth == 0 {
2025-12-04 13:42:32 +08:00
fm.logger.Warn("字体添加后验证失败测试文本宽度为0但会在Output时重新尝试",
zap.String("font_path", normalizedPath),
2025-12-04 13:20:03 +08:00
zap.String("font_name", fontName),
)
2025-12-04 13:42:32 +08:00
// 注意即使验证失败也返回true因为gofpdf在Output时才会真正加载字体文件
// 这里的验证可能不准确
2025-12-03 15:53:31 +08:00
}
2025-12-04 13:42:32 +08:00
fm.logger.Info("字体已添加到PDF将在Output时加载",
zap.String("font_path", normalizedPath),
2025-12-04 13:20:03 +08:00
zap.String("font_name", fontName),
zap.Float64("test_width", testWidth),
)
2025-12-03 12:03:42 +08:00
return true
}
2025-12-03 15:53:31 +08:00
// getChineseFontPaths 获取中文字体路径列表仅TTF格式
func (fm *FontManager) getChineseFontPaths() []string {
// 按优先级排序的字体文件列表
fontNames := []string{
"simhei.ttf", // 黑体(默认)
"simkai.ttf", // 楷体(备选)
"simfang.ttf", // 仿宋(备选)
2025-12-03 12:03:42 +08:00
}
2025-12-03 15:53:31 +08:00
return fm.buildFontPaths(fontNames)
}
// getWatermarkFontPaths 获取水印字体路径列表仅TTF格式
func (fm *FontManager) getWatermarkFontPaths() []string {
2025-12-04 12:56:39 +08:00
// 水印字体文件名(尝试大小写变体)
2025-12-03 15:53:31 +08:00
fontNames := []string{
2026-03-13 18:07:24 +08:00
// "XuanZongTi-v0.1.otf", //玄宗字体不支持otf
"WenYuanSerifSC-Bold.ttf", //文渊雅黑
// "YunFengFeiYunTi-2.ttf", // 毛笔字体
// "yunfengfeiyunti-2.ttf", // 毛笔字体小写版本(兼容)
2025-12-03 12:03:42 +08:00
}
2025-12-03 15:53:31 +08:00
return fm.buildFontPaths(fontNames)
2025-12-03 12:03:42 +08:00
}
2026-03-16 12:32:41 +08:00
// getBodyFontPaths 获取正文宋体路径列表(小四对应 12pt
// 优先使用 resources/pdf/fonts/simsun.ttc宋体
func (fm *FontManager) getBodyFontPaths() []string {
fontNames := []string{
// "simsun.ttc", // 宋体(项目内 resources/pdf/fonts
"simsun.ttf",
"SimSun.ttf",
"WenYuanSerifSC-Bold.ttf", // 文渊宋体风格,备选
}
return fm.buildFontPaths(fontNames)
}
2025-12-04 12:56:39 +08:00
// buildFontPaths 构建字体文件路径列表仅从resources/pdf/fonts加载
2025-12-04 13:42:32 +08:00
// 返回所有存在的字体文件的绝对路径
2025-12-03 15:53:31 +08:00
func (fm *FontManager) buildFontPaths(fontNames []string) []string {
2025-12-04 13:42:32 +08:00
// 获取resources/pdf目录已返回绝对路径
2025-12-04 12:56:39 +08:00
resourcesPDFDir := GetResourcesPDFDir()
2025-12-04 13:42:32 +08:00
if resourcesPDFDir == "" {
fm.logger.Error("无法获取resources/pdf目录路径")
return []string{}
2025-12-04 13:20:03 +08:00
}
2025-12-04 13:42:32 +08:00
// 构建字体目录路径resourcesPDFDir已经是绝对路径
fontsDir := filepath.Join(resourcesPDFDir, "fonts")
2025-12-04 13:20:03 +08:00
fm.logger.Debug("查找字体文件",
zap.String("resources_pdf_dir", resourcesPDFDir),
2025-12-04 13:42:32 +08:00
zap.String("fonts_dir", fontsDir),
2025-12-04 13:20:03 +08:00
zap.Strings("font_names", fontNames),
)
2025-12-04 13:42:32 +08:00
// 构建字体文件路径列表(都是绝对路径)
2025-12-04 13:20:03 +08:00
var fontPaths []string
2025-12-03 15:53:31 +08:00
for _, fontName := range fontNames {
2025-12-04 13:42:32 +08:00
fontPath := filepath.Join(fontsDir, fontName)
// 确保是绝对路径
if absPath, err := filepath.Abs(fontPath); err == nil {
fontPaths = append(fontPaths, absPath)
} else {
fm.logger.Warn("无法获取字体文件绝对路径",
2025-12-04 13:20:03 +08:00
zap.String("font_path", fontPath),
zap.Error(err),
)
}
2025-12-03 18:25:04 +08:00
}
2025-12-04 10:35:11 +08:00
// 过滤出实际存在的字体文件
2025-12-03 12:03:42 +08:00
var existingFonts []string
for _, fontPath := range fontPaths {
2025-12-04 13:42:32 +08:00
if info, err := os.Stat(fontPath); err == nil && info.Mode().IsRegular() {
2025-12-03 12:03:42 +08:00
existingFonts = append(existingFonts, fontPath)
2025-12-04 13:20:03 +08:00
fm.logger.Debug("找到字体文件", zap.String("font_path", fontPath))
} else {
fm.logger.Debug("字体文件不存在",
zap.String("font_path", fontPath),
zap.Error(err),
)
2025-12-03 12:03:42 +08:00
}
}
2025-12-04 13:20:03 +08:00
if len(existingFonts) == 0 {
fm.logger.Warn("未找到任何字体文件",
2025-12-04 13:42:32 +08:00
zap.String("fonts_dir", fontsDir),
2025-12-04 13:20:03 +08:00
zap.Strings("attempted_fonts", fontPaths),
)
} else {
fm.logger.Info("找到字体文件",
zap.Int("count", len(existingFonts)),
zap.Strings("font_paths", existingFonts),
)
}
2025-12-03 12:03:42 +08:00
return existingFonts
}
2025-12-03 15:53:31 +08:00
// SetFont 设置中文字体
2025-12-03 12:03:42 +08:00
func (fm *FontManager) SetFont(pdf *gofpdf.Fpdf, style string, size float64) {
if fm.chineseFontLoaded {
pdf.SetFont(fm.chineseFontName, style, size)
} else {
2025-12-03 15:53:31 +08:00
// 如果没有中文字体使用Arial作为后备
2025-12-03 12:03:42 +08:00
pdf.SetFont("Arial", style, size)
}
}
2025-12-03 15:53:31 +08:00
// SetWatermarkFont 设置水印字体
2025-12-03 12:03:42 +08:00
func (fm *FontManager) SetWatermarkFont(pdf *gofpdf.Fpdf, style string, size float64) {
if fm.watermarkFontLoaded {
pdf.SetFont(fm.watermarkFontName, style, size)
} else {
2025-12-03 15:53:31 +08:00
// 如果水印字体不可用,使用主字体作为后备
2025-12-03 12:03:42 +08:00
fm.SetFont(pdf, style, size)
}
}
2026-03-16 12:32:41 +08:00
// BodyFontSizeXiaosi 正文小四字号(约 12pt
const BodyFontSizeXiaosi = 12.0
// SetBodyFont 设置正文字体(宋体小四:描述、详情、说明、表格文字等)
func (fm *FontManager) SetBodyFont(pdf *gofpdf.Fpdf, style string, size float64) {
if size <= 0 {
size = BodyFontSizeXiaosi
}
if fm.bodyFontLoaded {
pdf.SetFont(fm.bodyFontName, style, size)
} else if fm.watermarkFontLoaded {
pdf.SetFont(fm.watermarkFontName, style, size)
} else {
fm.SetFont(pdf, style, size)
}
}
// IsBodyFontAvailable 正文字体(宋体)是否已加载
func (fm *FontManager) IsBodyFontAvailable() bool {
return fm.bodyFontLoaded || fm.watermarkFontLoaded
}
2025-12-03 12:03:42 +08:00
// IsChineseFontAvailable 检查中文字体是否可用
func (fm *FontManager) IsChineseFontAvailable() bool {
return fm.chineseFontLoaded
}