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

251 lines
6.7 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"
2025-12-03 12:03:42 +08:00
"runtime"
"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
2025-12-03 15:53:31 +08:00
baseDir string // 缓存基础目录,避免重复计算
2025-12-03 12:03:42 +08:00
}
// NewFontManager 创建字体管理器
func NewFontManager(logger *zap.Logger) *FontManager {
2025-12-03 15:53:31 +08:00
// 获取当前文件所在目录
_, filename, _, _ := runtime.Caller(0)
baseDir := filepath.Dir(filename)
2025-12-03 18:25:04 +08:00
fm := &FontManager{
2025-12-03 12:03:42 +08:00
logger: logger,
chineseFontName: "ChineseFont",
watermarkFontName: "WatermarkFont",
2025-12-03 15:53:31 +08:00
baseDir: baseDir,
2025-12-03 12:03:42 +08:00
}
2025-12-03 18:25:04 +08:00
return fm
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 10:35:11 +08:00
fm.logger.Warn("未找到中文字体文件")
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 10:35:11 +08:00
fm.logger.Warn("无法加载中文字体文件")
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 10:35:11 +08:00
// 静默处理,不记录日志
2025-12-03 12:03:42 +08:00
}
}()
2025-12-03 15:53:31 +08:00
// 检查文件是否存在
if _, err := os.Stat(fontPath); err != nil {
return false
}
2025-12-03 12:03:42 +08:00
2025-12-04 10:35:11 +08:00
// 将相对路径转换为绝对路径gofpdf在Output时需要绝对路径
absFontPath, err := filepath.Abs(fontPath)
if err != nil {
absFontPath = fontPath
}
// 再次检查绝对路径文件是否存在
if _, err := os.Stat(absFontPath); err != nil {
return false
}
// gofpdf v2使用AddUTF8Font添加支持UTF-8的字体使用绝对路径
pdf.AddUTF8Font(fontName, "", absFontPath) // 常规样式
pdf.AddUTF8Font(fontName, "B", absFontPath) // 粗体样式
2025-12-03 15:53:31 +08:00
// 验证字体是否可用
pdf.SetFont(fontName, "", 12)
testWidth := pdf.GetStringWidth("测试")
if testWidth == 0 {
return false
}
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 {
// 水印字体文件名(支持大小写变体)
fontNames := []string{
"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-03 18:25:04 +08:00
// buildFontPaths 构建字体文件路径列表(支持多种路径查找方式)
2025-12-03 15:53:31 +08:00
func (fm *FontManager) buildFontPaths(fontNames []string) []string {
2025-12-03 12:03:42 +08:00
var fontPaths []string
2025-12-04 10:35:11 +08:00
// 方式1: 优先使用相对路径Linux风格使用正斜杠- 最常用
for _, fontName := range fontNames {
// 相对路径(相对于项目根目录)
relativePaths := []string{
"internal/shared/pdf/fonts/" + fontName,
"./internal/shared/pdf/fonts/" + fontName,
}
fontPaths = append(fontPaths, relativePaths...)
}
// 方式2: 使用 pdf/fonts/ 目录(相对于当前文件)
2025-12-03 15:53:31 +08:00
for _, fontName := range fontNames {
fontPaths = append(fontPaths, filepath.Join(fm.baseDir, "fonts", fontName))
2025-12-03 12:03:42 +08:00
}
2025-12-04 10:35:11 +08:00
// 方式3: 尝试相对于工作目录的路径
2025-12-03 18:25:04 +08:00
if workDir, err := os.Getwd(); err == nil {
for _, fontName := range fontNames {
paths := []string{
filepath.Join(workDir, "internal", "shared", "pdf", "fonts", fontName),
filepath.Join(workDir, "tyapi-server", "internal", "shared", "pdf", "fonts", fontName),
}
fontPaths = append(fontPaths, paths...)
}
}
2025-12-04 10:35:11 +08:00
// 方式4: 尝试使用可执行文件所在目录
2025-12-03 18:25:04 +08:00
if execPath, err := os.Executable(); err == nil {
execDir := filepath.Dir(execPath)
if realPath, err := filepath.EvalSymlinks(execPath); err == nil {
execDir = filepath.Dir(realPath)
}
for _, fontName := range fontNames {
paths := []string{
filepath.Join(execDir, "internal", "shared", "pdf", "fonts", fontName),
filepath.Join(filepath.Dir(execDir), "internal", "shared", "pdf", "fonts", fontName),
}
fontPaths = append(fontPaths, paths...)
}
}
2025-12-04 10:35:11 +08:00
// 方式5: 尝试使用环境变量指定的路径
2025-12-03 18:25:04 +08:00
if fontDir := os.Getenv("PDF_FONT_DIR"); fontDir != "" {
for _, fontName := range fontNames {
fontPaths = append(fontPaths, filepath.Join(fontDir, fontName))
}
}
2025-12-04 10:35:11 +08:00
// 方式6: 服务器绝对路径(作为最后的后备方案)
if runtime.GOOS == "linux" {
commonServerPaths := []string{
"/www/tyapi-server/internal/shared/pdf/fonts",
"/app/internal/shared/pdf/fonts",
}
for _, basePath := range commonServerPaths {
for _, fontName := range fontNames {
fontPaths = append(fontPaths, filepath.Join(basePath, fontName))
}
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 10:35:11 +08:00
// 只记录关键错误
2025-12-03 18:25:04 +08:00
if len(existingFonts) == 0 {
2025-12-04 10:35:11 +08:00
fm.logger.Warn("未找到字体文件", zap.Strings("font_names", fontNames))
2025-12-03 18:25:04 +08:00
}
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
}