2025-12-03 12:03:42 +08:00
|
|
|
|
package pdf
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"os"
|
2025-12-03 15:53:31 +08:00
|
|
|
|
"path/filepath"
|
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
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 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",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
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
|
|
|
|
|
|
}
|
|
|
|
|
|
|
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:28:03 +08:00
|
|
|
|
// 再次确保是绝对路径(如果是相对路径,filepath.Abs可能返回错误)
|
|
|
|
|
|
if !filepath.IsAbs(absFontPath) {
|
|
|
|
|
|
if workDir, err := os.Getwd(); err == nil {
|
|
|
|
|
|
absFontPath = filepath.Join(workDir, absFontPath)
|
|
|
|
|
|
absFontPath, _ = filepath.Abs(absFontPath)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-04 13:20:03 +08:00
|
|
|
|
fm.logger.Debug("尝试添加字体",
|
2025-12-04 13:28:03 +08:00
|
|
|
|
zap.String("original_path", fontPath),
|
|
|
|
|
|
zap.String("absolute_path", absFontPath),
|
|
|
|
|
|
zap.Bool("is_absolute", filepath.IsAbs(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:28:03 +08:00
|
|
|
|
// 确保路径规范化(去除多余的路径组件)
|
|
|
|
|
|
absFontPath = filepath.Clean(absFontPath)
|
|
|
|
|
|
|
|
|
|
|
|
// gofpdf v2使用AddUTF8Font添加支持UTF-8的字体
|
|
|
|
|
|
// 重要:gofpdf 在 Output 时会重新访问字体文件
|
|
|
|
|
|
// 需要确保传入的路径是绝对路径,且路径分隔符使用系统正确的格式
|
|
|
|
|
|
// 在 Linux 下,路径应该以 / 开头
|
|
|
|
|
|
|
|
|
|
|
|
// 确保路径使用正确的分隔符(统一使用 /,gofpdf 内部会处理)
|
|
|
|
|
|
normalizedPath := filepath.ToSlash(absFontPath)
|
|
|
|
|
|
|
|
|
|
|
|
fm.logger.Debug("准备添加字体到gofpdf",
|
|
|
|
|
|
zap.String("abs_path", absFontPath),
|
|
|
|
|
|
zap.String("normalized_path", normalizedPath),
|
|
|
|
|
|
zap.String("font_name", fontName),
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
// AddUTF8Font 不返回错误,如果字体加载失败会在后续验证中发现
|
|
|
|
|
|
pdf.AddUTF8Font(fontName, "", normalizedPath) // 常规样式
|
|
|
|
|
|
pdf.AddUTF8Font(fontName, "B", normalizedPath) // 粗体样式
|
2025-12-03 15:53:31 +08:00
|
|
|
|
|
|
|
|
|
|
// 验证字体是否可用
|
|
|
|
|
|
pdf.SetFont(fontName, "", 12)
|
|
|
|
|
|
testWidth := pdf.GetStringWidth("测试")
|
|
|
|
|
|
if testWidth == 0 {
|
2025-12-04 13:20:03 +08:00
|
|
|
|
fm.logger.Warn("字体加载后验证失败(测试文本宽度为0)",
|
|
|
|
|
|
zap.String("font_path", absFontPath),
|
|
|
|
|
|
zap.String("font_name", fontName),
|
2025-12-04 13:28:03 +08:00
|
|
|
|
zap.Float64("test_width", testWidth),
|
2025-12-04 13:20:03 +08:00
|
|
|
|
)
|
2025-12-04 13:28:03 +08:00
|
|
|
|
// 注意:即使验证失败,也可能在后续使用中成功
|
|
|
|
|
|
// 因为 gofpdf 可能延迟加载字体
|
|
|
|
|
|
// 但为了安全,我们还是返回 false
|
2025-12-03 15:53:31 +08:00
|
|
|
|
return false
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-04 13:20:03 +08:00
|
|
|
|
fm.logger.Info("成功加载字体",
|
|
|
|
|
|
zap.String("font_path", absFontPath),
|
|
|
|
|
|
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{
|
2025-12-04 12:56:39 +08:00
|
|
|
|
"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
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-04 12:56:39 +08:00
|
|
|
|
|
|
|
|
|
|
// buildFontPaths 构建字体文件路径列表(仅从resources/pdf/fonts加载)
|
2025-12-03 15:53:31 +08:00
|
|
|
|
func (fm *FontManager) buildFontPaths(fontNames []string) []string {
|
2025-12-04 12:56:39 +08:00
|
|
|
|
// 获取resources/pdf目录(使用统一的资源路径查找函数)
|
|
|
|
|
|
resourcesPDFDir := GetResourcesPDFDir()
|
|
|
|
|
|
fontsDir := filepath.Join(resourcesPDFDir, "fonts")
|
2025-12-04 13:20:03 +08:00
|
|
|
|
|
|
|
|
|
|
// 确保fontsDir是绝对路径
|
|
|
|
|
|
absFontsDir, err := filepath.Abs(fontsDir)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
fm.logger.Warn("无法获取字体目录绝对路径",
|
|
|
|
|
|
zap.String("fonts_dir", fontsDir),
|
|
|
|
|
|
zap.Error(err),
|
|
|
|
|
|
)
|
|
|
|
|
|
absFontsDir = fontsDir
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fm.logger.Debug("查找字体文件",
|
|
|
|
|
|
zap.String("resources_pdf_dir", resourcesPDFDir),
|
|
|
|
|
|
zap.String("fonts_dir", absFontsDir),
|
|
|
|
|
|
zap.Strings("font_names", fontNames),
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
var fontPaths []string
|
2025-12-04 10:35:11 +08:00
|
|
|
|
|
2025-12-04 12:56:39 +08:00
|
|
|
|
// 只从resources/pdf/fonts目录加载
|
2025-12-03 15:53:31 +08:00
|
|
|
|
for _, fontName := range fontNames {
|
2025-12-04 13:20:03 +08:00
|
|
|
|
fontPath := filepath.Join(absFontsDir, fontName)
|
|
|
|
|
|
// 确保每个字体路径都是绝对路径
|
|
|
|
|
|
absFontPath, err := filepath.Abs(fontPath)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
fm.logger.Debug("无法获取字体文件绝对路径",
|
|
|
|
|
|
zap.String("font_path", fontPath),
|
|
|
|
|
|
zap.Error(err),
|
|
|
|
|
|
)
|
|
|
|
|
|
absFontPath = fontPath
|
|
|
|
|
|
}
|
|
|
|
|
|
fontPaths = append(fontPaths, absFontPath)
|
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-03 18:25:04 +08:00
|
|
|
|
fileInfo, err := os.Stat(fontPath)
|
2025-12-04 10:35:11 +08:00
|
|
|
|
if err == nil && fileInfo.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("未找到任何字体文件",
|
|
|
|
|
|
zap.String("fonts_dir", absFontsDir),
|
|
|
|
|
|
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)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// IsChineseFontAvailable 检查中文字体是否可用
|
|
|
|
|
|
func (fm *FontManager) IsChineseFontAvailable() bool {
|
|
|
|
|
|
return fm.chineseFontLoaded
|
|
|
|
|
|
}
|