Compare commits
	
		
			5 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 741611a2e1 | |||
| 133aeb8374 | |||
| b78a468632 | |||
| f1b4480e0f | |||
| ffffe4bf24 | 
							
								
								
									
										9
									
								
								ginext/jsonFilter.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								ginext/jsonFilter.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| package ginext | ||||
|  | ||||
| import "github.com/gin-gonic/gin" | ||||
|  | ||||
| var jsonFilterKey = "goext.jsonfilter" | ||||
|  | ||||
| func SetJSONFilter(g *gin.Context, filter string) { | ||||
| 	g.Set(jsonFilterKey, filter) | ||||
| } | ||||
| @@ -15,7 +15,7 @@ type jsonHTTPResponse struct { | ||||
|  | ||||
| func (j jsonHTTPResponse) jsonRenderer(g *gin.Context) json.GoJsonRender { | ||||
| 	var f *string | ||||
| 	if jsonfilter := g.GetString("goext.jsonfilter"); jsonfilter != "" { | ||||
| 	if jsonfilter := g.GetString(jsonFilterKey); jsonfilter != "" { | ||||
| 		f = &jsonfilter | ||||
| 	} | ||||
| 	return json.GoJsonRender{Data: j.data, NilSafeSlices: true, NilSafeMaps: true, Filter: f} | ||||
|   | ||||
| @@ -57,7 +57,7 @@ func (w *GinRoutesWrapper) Use(middleware ...gin.HandlerFunc) *GinRoutesWrapper | ||||
| } | ||||
|  | ||||
| func (w *GinRoutesWrapper) WithJSONFilter(filter string) *GinRoutesWrapper { | ||||
| 	return w.Use(func(g *gin.Context) { g.Set("goext.jsonfilter", filter) }) | ||||
| 	return w.Use(func(g *gin.Context) { g.Set(jsonFilterKey, filter) }) | ||||
| } | ||||
|  | ||||
| func (w *GinRoutesWrapper) GET(relativePath string) *GinRouteBuilder { | ||||
| @@ -112,7 +112,7 @@ func (w *GinRouteBuilder) Use(middleware ...gin.HandlerFunc) *GinRouteBuilder { | ||||
| } | ||||
|  | ||||
| func (w *GinRouteBuilder) WithJSONFilter(filter string) *GinRouteBuilder { | ||||
| 	return w.Use(func(g *gin.Context) { g.Set("goext.jsonfilter", filter) }) | ||||
| 	return w.Use(func(g *gin.Context) { g.Set(jsonFilterKey, filter) }) | ||||
| } | ||||
|  | ||||
| func (w *GinRouteBuilder) Handle(handler WHandlerFunc) { | ||||
|   | ||||
							
								
								
									
										18
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								go.mod
									
									
									
									
									
								
							| @@ -9,19 +9,19 @@ require ( | ||||
| 	github.com/rs/xid v1.5.0 | ||||
| 	github.com/rs/zerolog v1.33.0 | ||||
| 	go.mongodb.org/mongo-driver v1.16.0 | ||||
| 	golang.org/x/crypto v0.25.0 | ||||
| 	golang.org/x/sys v0.22.0 | ||||
| 	golang.org/x/term v0.22.0 | ||||
| 	golang.org/x/crypto v0.26.0 | ||||
| 	golang.org/x/sys v0.23.0 | ||||
| 	golang.org/x/term v0.23.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/disintegration/imaging v1.6.2 | ||||
| 	github.com/jung-kurt/gofpdf v1.16.2 | ||||
| 	golang.org/x/sync v0.7.0 | ||||
| 	golang.org/x/sync v0.8.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/bytedance/sonic v1.12.0 // indirect | ||||
| 	github.com/bytedance/sonic v1.12.1 // indirect | ||||
| 	github.com/bytedance/sonic/loader v0.2.0 // indirect | ||||
| 	github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect | ||||
| 	github.com/chenzhuoyu/iasm v0.9.1 // indirect | ||||
| @@ -53,10 +53,10 @@ require ( | ||||
| 	github.com/xdg-go/scram v1.1.2 // indirect | ||||
| 	github.com/xdg-go/stringprep v1.0.4 // indirect | ||||
| 	github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78 // indirect | ||||
| 	golang.org/x/arch v0.8.0 // indirect | ||||
| 	golang.org/x/image v0.18.0 // indirect | ||||
| 	golang.org/x/net v0.27.0 // indirect | ||||
| 	golang.org/x/text v0.16.0 // indirect | ||||
| 	golang.org/x/arch v0.9.0 // indirect | ||||
| 	golang.org/x/image v0.19.0 // indirect | ||||
| 	golang.org/x/net v0.28.0 // indirect | ||||
| 	golang.org/x/text v0.17.0 // indirect | ||||
| 	google.golang.org/protobuf v1.34.2 // indirect | ||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||
| 	modernc.org/libc v1.37.6 // indirect | ||||
|   | ||||
							
								
								
									
										18
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								go.sum
									
									
									
									
									
								
							| @@ -26,6 +26,8 @@ github.com/bytedance/sonic v1.11.9 h1:LFHENlIY/SLzDWverzdOvgMztTxcfcF+cqNsz9pK5z | ||||
| github.com/bytedance/sonic v1.11.9/go.mod h1:LysEHSvpvDySVdC2f87zGWf6CIKJcAvqab1ZaiQtds4= | ||||
| github.com/bytedance/sonic v1.12.0 h1:YGPgxF9xzaCNvd/ZKdQ28yRovhfMFZQjuk6fKBzZ3ls= | ||||
| github.com/bytedance/sonic v1.12.0/go.mod h1:B8Gt/XvtZ3Fqj+iSKMypzymZxw/FVwgIGKzMzT9r/rk= | ||||
| github.com/bytedance/sonic v1.12.1 h1:jWl5Qz1fy7X1ioY74WqO0KjAMtAGQs4sYnjiEBiyX24= | ||||
| github.com/bytedance/sonic v1.12.1/go.mod h1:B8Gt/XvtZ3Fqj+iSKMypzymZxw/FVwgIGKzMzT9r/rk= | ||||
| github.com/bytedance/sonic/loader v0.1.0/go.mod h1:UmRT+IRTGKz/DAkzcEGzyVqQFJ7H9BqwBO3pm9H/+HY= | ||||
| github.com/bytedance/sonic/loader v0.1.1 h1:c+e5Pt1k/cy5wMveRDyk2X4B9hF4g7an8N3zCYjJFNM= | ||||
| github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU= | ||||
| @@ -237,6 +239,8 @@ golang.org/x/arch v0.7.0 h1:pskyeJh/3AmoQ8CPE95vxHLqp1G1GfGNXTmcl9NEKTc= | ||||
| golang.org/x/arch v0.7.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys= | ||||
| golang.org/x/arch v0.8.0 h1:3wRIsP3pM4yUptoR96otTUOXI367OS0+c9eeRi9doIc= | ||||
| golang.org/x/arch v0.8.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys= | ||||
| golang.org/x/arch v0.9.0 h1:ub9TgUInamJ8mrZIGlBG6/4TqWeMszd4N8lNorbrr6k= | ||||
| golang.org/x/arch v0.9.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20200302210943-78000ba7a073/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||
| @@ -259,6 +263,8 @@ golang.org/x/crypto v0.24.0 h1:mnl8DM0o513X8fdIkmyFE/5hTYxbwYOjDS/+rK6qpRI= | ||||
| golang.org/x/crypto v0.24.0/go.mod h1:Z1PMYSOR5nyMcyAVAIQSKCDwalqy85Aqn1x3Ws4L5DM= | ||||
| golang.org/x/crypto v0.25.0 h1:ypSNr+bnYL2YhwoMt2zPxHFmbAN1KZs/njMG3hxUp30= | ||||
| golang.org/x/crypto v0.25.0/go.mod h1:T+wALwcMOSE0kXgUAnPAHqTLW+XHgcELELW8VaDgm/M= | ||||
| golang.org/x/crypto v0.26.0 h1:RrRspgV4mU+YwB4FYnuBoKsUapNIL5cohGAmSH3azsw= | ||||
| golang.org/x/crypto v0.26.0/go.mod h1:GY7jblb9wI+FOo5y8/S2oY4zWP07AkOJ4+jxCqdqn54= | ||||
| golang.org/x/image v0.0.0-20190910094157-69e4b8554b2a/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||
| golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8 h1:hVwzHzIUGRjiF7EcUjqNxk3NCfkPxbDKRdnNE1Rpg0U= | ||||
| golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||
| @@ -268,6 +274,8 @@ golang.org/x/image v0.17.0 h1:nTRVVdajgB8zCMZVsViyzhnMKPwYeroEERRC64JuLco= | ||||
| golang.org/x/image v0.17.0/go.mod h1:4yyo5vMFQjVjUcVk4jEQcU9MGy/rulF5WvUILseCM2E= | ||||
| golang.org/x/image v0.18.0 h1:jGzIakQa/ZXI1I0Fxvaa9W7yP25TqT6cHIHn+6CqvSQ= | ||||
| golang.org/x/image v0.18.0/go.mod h1:4yyo5vMFQjVjUcVk4jEQcU9MGy/rulF5WvUILseCM2E= | ||||
| golang.org/x/image v0.19.0 h1:D9FX4QWkLfkeqaC62SonffIIuYdOk/UE2XKUBgRIBIQ= | ||||
| golang.org/x/image v0.19.0/go.mod h1:y0zrRqlQRWQ5PXaYCOMLTW2fpsxZ8Qh9I/ohnInJEys= | ||||
| golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| @@ -290,12 +298,16 @@ golang.org/x/net v0.26.0 h1:soB7SVo0PWrY4vPW/+ay0jKDNScG2X9wFeYlXIvJsOQ= | ||||
| golang.org/x/net v0.26.0/go.mod h1:5YKkiSynbBIh3p6iOc/vibscux0x38BZDkn8sCUPxHE= | ||||
| golang.org/x/net v0.27.0 h1:5K3Njcw06/l2y9vpGCSdcxWOYHOUk3dVNGDXN+FvAys= | ||||
| golang.org/x/net v0.27.0/go.mod h1:dDi0PyhWNoiUOrAS8uXv/vnScO4wnHQO4mj9fn/RytE= | ||||
| golang.org/x/net v0.28.0 h1:a9JDOJc5GMUJ0+UDqmLT86WiEy7iWyIhz8gz8E4e5hE= | ||||
| golang.org/x/net v0.28.0/go.mod h1:yqtgsTWOOnlGLG9GFRrK3++bGOUEkNBoHZc8MEDWPNg= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.6.0 h1:5BMeUDZ7vkXGfEr1x9B4bRcTH4lpkTkpdh0T/J+qjbQ= | ||||
| golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sync v0.7.0 h1:YsImfSBoP9QPYL0xyKJPq0gcaJdG3rInoqxTWbfQu9M= | ||||
| golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ= | ||||
| golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| @@ -321,6 +333,8 @@ golang.org/x/sys v0.21.0 h1:rF+pYz3DAGSQAxAu1CbC7catZg4ebC4UIeIhKxBZvws= | ||||
| golang.org/x/sys v0.21.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/sys v0.22.0 h1:RI27ohtqKCnwULzJLqkv897zojh5/DwS/ENaMzUOaWI= | ||||
| golang.org/x/sys v0.22.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/sys v0.23.0 h1:YfKFowiIMvtgl1UERQoTPPToxltDeZfbj4H7dVUCwmM= | ||||
| golang.org/x/sys v0.23.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||
| golang.org/x/term v0.16.0 h1:m+B6fahuftsE9qjo0VWp2FW0mB3MTJvR0BaMQrq0pmE= | ||||
| @@ -337,6 +351,8 @@ golang.org/x/term v0.21.0 h1:WVXCp+/EBEHOj53Rvu+7KiT/iElMrO8ACK16SMZ3jaA= | ||||
| golang.org/x/term v0.21.0/go.mod h1:ooXLefLobQVslOqselCNF4SxFAaoS6KujMbsGzSDmX0= | ||||
| golang.org/x/term v0.22.0 h1:BbsgPEJULsl2fV/AT3v15Mjva5yXKQDyKf+TbDz7QJk= | ||||
| golang.org/x/term v0.22.0/go.mod h1:F3qCibpT5AMpCRfhfT53vVJwhLtIVHhB9XDjfFvnMI4= | ||||
| golang.org/x/term v0.23.0 h1:F6D4vR+EHoL9/sWAWgAR1H2DcHr4PareCbAaCo1RpuU= | ||||
| golang.org/x/term v0.23.0/go.mod h1:DgV24QBUrK6jhZXl+20l6UWznPlwAHm1Q1mGHtydmSk= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| @@ -349,6 +365,8 @@ golang.org/x/text v0.15.0 h1:h1V/4gjBv8v9cjcR6+AR5+/cIYK5N/WAgiv4xlsEtAk= | ||||
| golang.org/x/text v0.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= | ||||
| golang.org/x/text v0.16.0 h1:a94ExnEXNtEwYLGJSIUxnWoxoRz/ZcCsV63ROupILh4= | ||||
| golang.org/x/text v0.16.0/go.mod h1:GhwF1Be+LQoKShO3cGOHzqOgRrGaYc9AvblQOmPVHnI= | ||||
| golang.org/x/text v0.17.0 h1:XtiM5bkSOt+ewxlOE/aE/AKEHibwj/6gvWMl9Rsh0Qc= | ||||
| golang.org/x/text v0.17.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| package goext | ||||
|  | ||||
| const GoextVersion = "0.0.491" | ||||
| const GoextVersion = "0.0.496" | ||||
|  | ||||
| const GoextVersionTimestamp = "2024-07-31T00:15:09+0200" | ||||
| const GoextVersionTimestamp = "2024-08-07T15:34:06+0200" | ||||
|   | ||||
| @@ -788,7 +788,7 @@ FieldLoop: | ||||
|  | ||||
| 		if f.omitEmpty && isEmptyValue(fv) { | ||||
| 			continue | ||||
| 		} else if opts.filter != nil && len(f.jsonfilter) > 0 && !f.jsonfilter.Contains(*opts.filter) { | ||||
| 		} else if opts.filter != nil && !matchesJSONFilter(f.jsonfilter, *opts.filter) { | ||||
| 			continue | ||||
| 		} | ||||
| 		e.WriteByte(next) | ||||
| @@ -808,6 +808,26 @@ FieldLoop: | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func matchesJSONFilter(filter jsonfilter, value string) bool { | ||||
| 	if len(filter) == 0 { | ||||
| 		return true | ||||
| 	} | ||||
|  | ||||
| 	if len(filter) == 1 && filter[0] == "-" { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	if filter.Contains(value) { | ||||
| 		return true | ||||
| 	} | ||||
|  | ||||
| 	if filter.Contains("*") { | ||||
| 		return true | ||||
| 	} | ||||
|  | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| func newStructEncoder(t reflect.Type, tagkey string) encoderFunc { | ||||
| 	se := structEncoder{fields: cachedTypeFields(t, tagkey)} | ||||
| 	return se.encode | ||||
|   | ||||
| @@ -169,7 +169,7 @@ func EncodeImage(img image.Image, compression ImageCompresson) (bytes.Buffer, st | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ObjectFitImage(img image.Image, bbw float64, bbh float64, fit ImageFit, fillColor color.Color) (image.Image, error) { | ||||
| func ObjectFitImage(img image.Image, bbw float64, bbh float64, fit ImageFit, fillColor color.Color) (image.Image, PercentageRectangle, error) { | ||||
|  | ||||
| 	iw := img.Bounds().Size().X | ||||
| 	ih := img.Bounds().Size().Y | ||||
| @@ -214,12 +214,12 @@ func ObjectFitImage(img image.Image, bbw float64, bbh float64, fit ImageFit, fil | ||||
| 		draw.Draw(newImg, newImg.Bounds(), &image.Uniform{C: fillColor}, image.Pt(0, 0), draw.Src) | ||||
| 		draw.Draw(newImg, newImg.Bounds(), img, image.Pt(0, 0), draw.Over) | ||||
|  | ||||
| 		return newImg, nil | ||||
| 		return newImg, PercentageRectangle{0, 0, 1, 1}, nil | ||||
| 	} | ||||
|  | ||||
| 	if fit == ImageFitContainCenter || fit == ImageFitContainTopLeft || fit == ImageFitContainTopRight || fit == ImageFitContainBottomLeft || fit == ImageFitContainBottomRight { | ||||
|  | ||||
| 		// image-fit:cover fills the target-bounding-box with the image, there is potentially empty-space, it potentially cuts parts of the image away | ||||
| 		// image-fit:contain fills the target-bounding-box with the image, there is potentially empty-space, it potentially cuts parts of the image away | ||||
|  | ||||
| 		// we use the bigger (!) value of facW and facH, | ||||
| 		// because the image is made to fit the bounding-box, the bigger factor (= the dimension the image is stretched less) is relevant | ||||
| @@ -266,7 +266,7 @@ func ObjectFitImage(img image.Image, bbw float64, bbh float64, fit ImageFit, fil | ||||
| 		draw.Draw(newImg, newImg.Bounds(), &image.Uniform{C: fillColor}, image.Pt(0, 0), draw.Src) | ||||
| 		draw.Draw(newImg, destBounds, img, image.Pt(0, 0), draw.Over) | ||||
|  | ||||
| 		return newImg, nil | ||||
| 		return newImg, calcRelativeRect(destBounds, newImg.Bounds()), nil | ||||
| 	} | ||||
|  | ||||
| 	if fit == ImageFitStretch { | ||||
| @@ -293,10 +293,10 @@ func ObjectFitImage(img image.Image, bbw float64, bbh float64, fit ImageFit, fil | ||||
| 		draw.Draw(newImg, newImg.Bounds(), &image.Uniform{C: fillColor}, image.Pt(0, 0), draw.Src) | ||||
| 		draw.Draw(newImg, newImg.Bounds(), img, image.Pt(0, 0), draw.Over) | ||||
|  | ||||
| 		return newImg, nil | ||||
| 		return newImg, PercentageRectangle{0, 0, 1, 1}, nil | ||||
| 	} | ||||
|  | ||||
| 	return nil, exerr.New(exerr.TypeInternal, fmt.Sprintf("unknown image-fit: '%s'", fit)).Build() | ||||
| 	return nil, PercentageRectangle{}, exerr.New(exerr.TypeInternal, fmt.Sprintf("unknown image-fit: '%s'", fit)).Build() | ||||
| } | ||||
|  | ||||
| func VerifyAndDecodeImage(data io.Reader, mime string) (image.Image, error) { | ||||
|   | ||||
							
								
								
									
										35
									
								
								imageext/types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								imageext/types.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| package imageext | ||||
|  | ||||
| import "image" | ||||
|  | ||||
| type Rectangle struct { | ||||
| 	X float64 | ||||
| 	Y float64 | ||||
| 	W float64 | ||||
| 	H float64 | ||||
| } | ||||
|  | ||||
| type PercentageRectangle struct { | ||||
| 	X float64 // [0..1] | ||||
| 	Y float64 // [0..1] | ||||
| 	W float64 // [0..1] | ||||
| 	H float64 // [0..1] | ||||
| } | ||||
|  | ||||
| func (r PercentageRectangle) Of(ref Rectangle) Rectangle { | ||||
| 	return Rectangle{ | ||||
| 		X: ref.X + r.X*ref.W, | ||||
| 		Y: ref.Y + r.Y*ref.H, | ||||
| 		W: r.W * ref.W, | ||||
| 		H: r.H * ref.H, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func calcRelativeRect(inner image.Rectangle, outer image.Rectangle) PercentageRectangle { | ||||
| 	return PercentageRectangle{ | ||||
| 		X: float64(inner.Min.X-outer.Min.X) / float64(outer.Dx()), | ||||
| 		Y: float64(inner.Min.Y-outer.Min.Y) / float64(outer.Dy()), | ||||
| 		W: float64(inner.Dx()) / float64(outer.Dx()), | ||||
| 		H: float64(inner.Dy()) / float64(outer.Dy()), | ||||
| 	} | ||||
| } | ||||
| @@ -323,6 +323,16 @@ func ArrMap[T1 any, T2 any](arr []T1, conv func(v T1) T2) []T2 { | ||||
| 	return r | ||||
| } | ||||
|  | ||||
| func ArrDeRef[T1 any](arr []*T1) []T1 { | ||||
| 	r := make([]T1, 0, len(arr)) | ||||
| 	for _, v := range arr { | ||||
| 		if v != nil { | ||||
| 			r = append(r, *v) | ||||
| 		} | ||||
| 	} | ||||
| 	return r | ||||
| } | ||||
|  | ||||
| func MapMap[TK comparable, TV any, TR any](inmap map[TK]TV, conv func(k TK, v TV) TR) []TR { | ||||
| 	r := make([]TR, 0, len(inmap)) | ||||
| 	for k, v := range inmap { | ||||
|   | ||||
							
								
								
									
										1
									
								
								wpdf/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								wpdf/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| wpdf_test.pdf | ||||
							
								
								
									
										
											BIN
										
									
								
								wpdf/logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								wpdf/logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 11 KiB | 
							
								
								
									
										81
									
								
								wpdf/wpdf.go
									
									
									
									
									
								
							
							
						
						
									
										81
									
								
								wpdf/wpdf.go
									
									
									
									
									
								
							| @@ -3,6 +3,7 @@ package wpdf | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"github.com/jung-kurt/gofpdf" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| ) | ||||
|  | ||||
| type WPDFBuilder struct { | ||||
| @@ -13,6 +14,7 @@ type WPDFBuilder struct { | ||||
| 	fontName    PDFFontFamily | ||||
| 	fontStyle   PDFFontStyle | ||||
| 	fontSize    float64 | ||||
| 	debug       bool | ||||
| } | ||||
|  | ||||
| type PDFMargins struct { | ||||
| @@ -61,6 +63,19 @@ func (b *WPDFBuilder) SetMargins(v PDFMargins) { | ||||
|  | ||||
| func (b *WPDFBuilder) AddPage() { | ||||
| 	b.b.AddPage() | ||||
|  | ||||
| 	if b.debug { | ||||
|  | ||||
| 		ml, mt, mr, mb := b.GetMargins() | ||||
| 		pw, ph := b.GetPageSize() | ||||
|  | ||||
| 		b.Rect(pw-ml-mr, ph-mt-mb, RectOutline, NewPDFRectOpt().X(ml).Y(mt).LineWidth(0.25).DrawColor(0, 0, 128)) | ||||
|  | ||||
| 		b.Rect(pw, mt, RectFill, NewPDFRectOpt().X(0).Y(0).FillColor(0, 0, 255).Alpha(0.2, BlendNormal)) | ||||
| 		b.Rect(ml, ph-mt-mb, RectFill, NewPDFRectOpt().X(0).Y(mt).FillColor(0, 0, 255).Alpha(0.2, BlendNormal)) | ||||
| 		b.Rect(mr, ph-mt-mb, RectFill, NewPDFRectOpt().X(pw-mr).Y(mt).FillColor(0, 0, 255).Alpha(0.2, BlendNormal)) | ||||
| 		b.Rect(pw, mb, RectFill, NewPDFRectOpt().X(0).Y(ph-mb).FillColor(0, 0, 255).Alpha(0.2, BlendNormal)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) SetTextColor(cr, cg, cb int) { | ||||
| @@ -105,12 +120,38 @@ func (b *WPDFBuilder) SetFont(fontName PDFFontFamily, fontStyle PDFFontStyle, fo | ||||
| 	b.cellHeight = b.b.PointConvert(fontSize) | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) GetFontSize() float64 { | ||||
| 	return b.fontSize | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) GetFontFamily() PDFFontStyle { | ||||
| 	return b.fontStyle | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) GetFontStyle() float64 { | ||||
| 	return b.fontSize | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) SetCellSpacing(h float64) { | ||||
| 	b.cellSpacing = h | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) Ln(h float64) { | ||||
| 	xBefore, yBefore := b.GetXY() | ||||
|  | ||||
| 	b.b.Ln(h) | ||||
|  | ||||
| 	yAfter := b.GetY() | ||||
|  | ||||
| 	if b.debug { | ||||
|  | ||||
| 		_, _, mr, _ := b.GetMargins() | ||||
| 		pw, _ := b.GetPageSize() | ||||
|  | ||||
| 		b.Rect(pw-mr-xBefore, yAfter-yBefore, RectOutline, NewPDFRectOpt().X(xBefore).Y(yBefore).LineWidth(0.25).DrawColor(128, 128, 0).Alpha(0.5, BlendNormal)) | ||||
| 		b.Rect(pw-mr-xBefore, yAfter-yBefore, RectFill, NewPDFRectOpt().X(xBefore).Y(yBefore).LineWidth(0.25).FillColor(128, 128, 0).Alpha(0.1, BlendNormal)) | ||||
| 		b.Line(xBefore, yBefore, pw-mr, yAfter, NewPDFLineOpt().LineWidth(0.25).DrawColor(128, 128, 0)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) Build() ([]byte, error) { | ||||
| @@ -192,6 +233,44 @@ func (b *WPDFBuilder) GetWorkAreaWidth() float64 { | ||||
| 	return b.GetPageWidth() - b.GetMarginLeft() - b.GetMarginRight() | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) GetStringWidth(str string) float64 { | ||||
| func (b *WPDFBuilder) SetAutoPageBreak(auto bool, margin float64) { | ||||
| 	b.b.SetAutoPageBreak(auto, margin) | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) SetFooterFunc(fnc func()) { | ||||
| 	b.b.SetFooterFunc(fnc) | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) PageNo() int { | ||||
| 	return b.b.PageNo() | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) GetStringWidth(str string, opts ...PDFCellOpt) float64 { | ||||
|  | ||||
| 	var fontNameOverride *PDFFontFamily | ||||
| 	var fontStyleOverride *PDFFontStyle | ||||
| 	var fontSizeOverride *float64 | ||||
|  | ||||
| 	for _, opt := range opts { | ||||
| 		fontNameOverride = langext.CoalesceOpt(opt.fontNameOverride, fontNameOverride) | ||||
| 		fontStyleOverride = langext.CoalesceOpt(opt.fontStyleOverride, fontStyleOverride) | ||||
| 		fontSizeOverride = langext.CoalesceOpt(opt.fontSizeOverride, fontSizeOverride) | ||||
| 	} | ||||
|  | ||||
| 	if fontNameOverride != nil || fontStyleOverride != nil || fontSizeOverride != nil { | ||||
| 		oldFontName := b.fontName | ||||
| 		oldFontStyle := b.fontStyle | ||||
| 		oldFontSize := b.fontSize | ||||
| 		newFontName := langext.Coalesce(fontNameOverride, oldFontName) | ||||
| 		newFontStyle := langext.Coalesce(fontStyleOverride, oldFontStyle) | ||||
| 		newFontSize := langext.Coalesce(fontSizeOverride, oldFontSize) | ||||
| 		b.SetFont(newFontName, newFontStyle, newFontSize) | ||||
| 		defer func() { b.SetFont(oldFontName, oldFontStyle, oldFontSize) }() | ||||
| 	} | ||||
|  | ||||
| 	return b.b.GetStringWidth(str) | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) Debug(v bool) { | ||||
| 	b.debug = v | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,9 @@ | ||||
| package wpdf | ||||
|  | ||||
| import "gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| import ( | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| ) | ||||
|  | ||||
| type PDFCellOpt struct { | ||||
| 	width             *float64 | ||||
| @@ -14,6 +17,7 @@ type PDFCellOpt struct { | ||||
| 	fontNameOverride  *PDFFontFamily | ||||
| 	fontStyleOverride *PDFFontStyle | ||||
| 	fontSizeOverride  *float64 | ||||
| 	alphaOverride     *dataext.Tuple[float64, PDFBlendMode] | ||||
| 	extraLn           *float64 | ||||
| 	x                 *float64 | ||||
| 	autoWidth         *bool | ||||
| @@ -21,6 +25,7 @@ type PDFCellOpt struct { | ||||
| 	borderColor       *PDFColor | ||||
| 	fillColor         *PDFColor | ||||
| 	autoWidthPaddingX *float64 | ||||
| 	debug             *bool | ||||
| } | ||||
|  | ||||
| func NewPDFCellOpt() *PDFCellOpt { | ||||
| @@ -149,12 +154,45 @@ func (opt *PDFCellOpt) FillColorHex(c uint32) *PDFCellOpt { | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFCellOpt) Alpha(alpha float64, blendMode PDFBlendMode) *PDFCellOpt { | ||||
| 	opt.alphaOverride = &dataext.Tuple[float64, PDFBlendMode]{V1: alpha, V2: blendMode} | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFCellOpt) Debug(v bool) *PDFCellOpt { | ||||
| 	opt.debug = &v | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFCellOpt) Copy() *PDFCellOpt { | ||||
| 	c := *opt | ||||
| 	return &c | ||||
| } | ||||
|  | ||||
| func (opt *PDFCellOpt) ToMulti() *PDFMultiCellOpt { | ||||
| 	return &PDFMultiCellOpt{ | ||||
| 		width:             opt.width, | ||||
| 		height:            opt.height, | ||||
| 		border:            opt.border, | ||||
| 		align:             opt.align, | ||||
| 		fill:              opt.fill, | ||||
| 		fontNameOverride:  opt.fontNameOverride, | ||||
| 		fontStyleOverride: opt.fontStyleOverride, | ||||
| 		fontSizeOverride:  opt.fontSizeOverride, | ||||
| 		extraLn:           opt.extraLn, | ||||
| 		x:                 opt.x, | ||||
| 		textColor:         opt.textColor, | ||||
| 		borderColor:       opt.borderColor, | ||||
| 		fillColor:         opt.fillColor, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | ||||
|  | ||||
| 	txtTR := b.tr(txt) | ||||
|  | ||||
| 	width := float64(0) | ||||
| 	height := b.cellHeight + b.cellSpacing | ||||
| 	var height *float64 = nil | ||||
| 	border := BorderNone | ||||
| 	ln := BreakToNextLine | ||||
| 	align := AlignLeft | ||||
| @@ -164,6 +202,7 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | ||||
| 	var fontNameOverride *PDFFontFamily | ||||
| 	var fontStyleOverride *PDFFontStyle | ||||
| 	var fontSizeOverride *float64 | ||||
| 	var alphaOverride *dataext.Tuple[float64, PDFBlendMode] | ||||
| 	extraLn := float64(0) | ||||
| 	var x *float64 | ||||
| 	autoWidth := false | ||||
| @@ -171,10 +210,11 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | ||||
| 	var borderColor *PDFColor | ||||
| 	var fillColor *PDFColor | ||||
| 	autoWidthPaddingX := float64(0) | ||||
| 	debug := b.debug | ||||
|  | ||||
| 	for _, opt := range opts { | ||||
| 		width = langext.Coalesce(opt.width, width) | ||||
| 		height = langext.Coalesce(opt.height, height) | ||||
| 		height = langext.CoalesceOpt(opt.height, height) | ||||
| 		border = langext.Coalesce(opt.border, border) | ||||
| 		ln = langext.Coalesce(opt.ln, ln) | ||||
| 		align = langext.Coalesce(opt.align, align) | ||||
| @@ -184,6 +224,7 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | ||||
| 		fontNameOverride = langext.CoalesceOpt(opt.fontNameOverride, fontNameOverride) | ||||
| 		fontStyleOverride = langext.CoalesceOpt(opt.fontStyleOverride, fontStyleOverride) | ||||
| 		fontSizeOverride = langext.CoalesceOpt(opt.fontSizeOverride, fontSizeOverride) | ||||
| 		alphaOverride = langext.CoalesceOpt(opt.alphaOverride, alphaOverride) | ||||
| 		extraLn = langext.Coalesce(opt.extraLn, extraLn) | ||||
| 		x = langext.CoalesceOpt(opt.x, x) | ||||
| 		autoWidth = langext.Coalesce(opt.autoWidth, autoWidth) | ||||
| @@ -191,6 +232,7 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | ||||
| 		borderColor = langext.CoalesceOpt(opt.borderColor, borderColor) | ||||
| 		fillColor = langext.CoalesceOpt(opt.fillColor, fillColor) | ||||
| 		autoWidthPaddingX = langext.Coalesce(opt.autoWidthPaddingX, autoWidthPaddingX) | ||||
| 		debug = langext.Coalesce(opt.debug, debug) | ||||
| 	} | ||||
|  | ||||
| 	if fontNameOverride != nil || fontStyleOverride != nil || fontSizeOverride != nil { | ||||
| @@ -204,6 +246,11 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | ||||
| 		defer func() { b.SetFont(oldFontName, oldFontStyle, oldFontSize) }() | ||||
| 	} | ||||
|  | ||||
| 	if height == nil { | ||||
| 		// (do after SetFont, so that b.cellHeight is correctly set to fontOverride) | ||||
| 		height = langext.Ptr(b.cellHeight + b.cellSpacing) | ||||
| 	} | ||||
|  | ||||
| 	if textColor != nil { | ||||
| 		oldColorR, oldColorG, oldColorB := b.b.GetTextColor() | ||||
| 		b.SetTextColor(textColor.R, textColor.G, textColor.B) | ||||
| @@ -222,15 +269,33 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | ||||
| 		defer func() { b.SetFillColor(oldColorR, oldColorG, oldColorB) }() | ||||
| 	} | ||||
|  | ||||
| 	if alphaOverride != nil { | ||||
| 		oldA, oldBMS := b.b.GetAlpha() | ||||
| 		b.b.SetAlpha(alphaOverride.V1, string(alphaOverride.V2)) | ||||
| 		defer func() { b.b.SetAlpha(oldA, oldBMS) }() | ||||
| 	} | ||||
|  | ||||
| 	if x != nil { | ||||
| 		b.b.SetX(*x) | ||||
| 	} | ||||
|  | ||||
| 	if autoWidth { | ||||
| 		width = b.b.GetStringWidth(txtTR) + autoWidthPaddingX | ||||
| 		width = b.GetStringWidth(txtTR, langext.ArrDeRef(opts)...) + autoWidthPaddingX | ||||
| 	} | ||||
|  | ||||
| 	b.b.CellFormat(width, height, txtTR, string(border), int(ln), string(align), fill, link, linkStr) | ||||
| 	xBefore, yBefore := b.b.GetXY() | ||||
|  | ||||
| 	b.b.CellFormat(width, *height, txtTR, string(border), int(ln), string(align), fill, link, linkStr) | ||||
|  | ||||
| 	if debug { | ||||
| 		if ln == BreakToNextLine { | ||||
| 			b.Rect(b.GetPageWidth()-xBefore-b.GetMarginRight(), *height, RectOutline, NewPDFRectOpt().X(xBefore).Y(yBefore).LineWidth(0.25).DrawColor(0, 128, 0)) | ||||
| 		} else if ln == BreakToRight { | ||||
| 			b.Rect(b.GetX()-xBefore, *height, RectOutline, NewPDFRectOpt().X(xBefore).Y(yBefore).LineWidth(0.25).DrawColor(0, 128, 0)) | ||||
| 		} else if ln == BreakToBelow { | ||||
| 			b.Rect(b.GetPageWidth(), *height, RectOutline, NewPDFRectOpt().X(xBefore).Y(yBefore).LineWidth(0.25).DrawColor(0, 128, 0)) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if extraLn != 0 { | ||||
| 		b.b.Ln(extraLn) | ||||
|   | ||||
| @@ -74,6 +74,35 @@ const ( | ||||
| 	RectFillOutline PDFRectStyle = "FD" | ||||
| ) | ||||
|  | ||||
| type PDFBlendMode string | ||||
|  | ||||
| const ( | ||||
| 	BlendNormal     PDFBlendMode = "Normal" | ||||
| 	BlendMultiply   PDFBlendMode = "Multiply" | ||||
| 	BlendScreen     PDFBlendMode = "Screen" | ||||
| 	BlendOverlay    PDFBlendMode = "Overlay" | ||||
| 	BlendDarken     PDFBlendMode = "Darken" | ||||
| 	BlendLighten    PDFBlendMode = "Lighten" | ||||
| 	BlendColorDodge PDFBlendMode = "ColorDodge" | ||||
| 	BlendColorBurn  PDFBlendMode = "ColorBurn" | ||||
| 	BlendHardLight  PDFBlendMode = "HardLight" | ||||
| 	BlendSoftLight  PDFBlendMode = "SoftLight" | ||||
| 	BlendDifference PDFBlendMode = "Difference" | ||||
| 	BlendExclusion  PDFBlendMode = "Exclusion" | ||||
| 	BlendHue        PDFBlendMode = "Hue" | ||||
| 	BlendSaturation PDFBlendMode = "Saturation" | ||||
| 	BlendColor      PDFBlendMode = "Color" | ||||
| 	BlendLuminosity PDFBlendMode = "Luminosity" | ||||
| ) | ||||
|  | ||||
| type PDFLineCapStyle string | ||||
|  | ||||
| const ( | ||||
| 	CapButt   PDFLineCapStyle = "butt" | ||||
| 	CapRound  PDFLineCapStyle = "round" | ||||
| 	CapSquare PDFLineCapStyle = "square" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	BackgroundFill        = true | ||||
| 	BackgroundTransparent = false | ||||
|   | ||||
| @@ -3,6 +3,7 @@ package wpdf | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"github.com/jung-kurt/gofpdf" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/imageext" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| 	"image" | ||||
| @@ -65,7 +66,12 @@ func (b *WPDFBuilder) RegisterImage(bin []byte, opts ...*PDFImageRegisterOpt) *P | ||||
| 	} | ||||
|  | ||||
| 	if imageType == "" { | ||||
| 		ct := http.DetectContentType(bin[:512]) | ||||
| 		ct := "" | ||||
| 		if len(bin) > 512 { | ||||
| 			ct = http.DetectContentType(bin[:512]) | ||||
| 		} else { | ||||
| 			ct = http.DetectContentType(bin) | ||||
| 		} | ||||
| 		switch ct { | ||||
| 		case "image/jpg": | ||||
| 			imageType = "JPG" | ||||
| @@ -125,6 +131,8 @@ type PDFImageOpt struct { | ||||
| 	compression           *imageext.ImageCompresson | ||||
| 	reEncodePixelPerMM    *float64 | ||||
| 	crop                  *imageext.ImageCrop | ||||
| 	alphaOverride         *dataext.Tuple[float64, PDFBlendMode] | ||||
| 	debug                 *bool | ||||
| } | ||||
|  | ||||
| func NewPDFImageOpt() *PDFImageOpt { | ||||
| @@ -151,6 +159,11 @@ func (opt *PDFImageOpt) Height(v float64) *PDFImageOpt { | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFImageOpt) Debug(v bool) *PDFImageOpt { | ||||
| 	opt.debug = &v | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFImageOpt) Flow(v bool) *PDFImageOpt { | ||||
| 	opt.flow = &v | ||||
| 	return opt | ||||
| @@ -212,6 +225,11 @@ func (opt *PDFImageOpt) Crop(cropX float64, cropY float64, cropWidth float64, cr | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFImageOpt) Alpha(alpha float64, blendMode PDFBlendMode) *PDFImageOpt { | ||||
| 	opt.alphaOverride = &dataext.Tuple[float64, PDFBlendMode]{V1: alpha, V2: blendMode} | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | ||||
| 	var err error | ||||
|  | ||||
| @@ -229,7 +247,9 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | ||||
| 	var imageFit *imageext.ImageFit = nil | ||||
| 	var fillColor color.Color = color.Transparent | ||||
| 	compression := imageext.CompressionPNGSpeed | ||||
| 	debug := b.debug | ||||
| 	var crop *imageext.ImageCrop = nil | ||||
| 	var alphaOverride *dataext.Tuple[float64, PDFBlendMode] | ||||
|  | ||||
| 	for _, opt := range opts { | ||||
| 		x = langext.Coalesce(opt.x, x) | ||||
| @@ -247,10 +267,18 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | ||||
| 		compression = langext.Coalesce(opt.compression, compression) | ||||
| 		reEncodePixelPerMM = langext.Coalesce(opt.reEncodePixelPerMM, reEncodePixelPerMM) | ||||
| 		crop = langext.CoalesceOpt(opt.crop, crop) | ||||
| 		debug = langext.Coalesce(opt.debug, debug) | ||||
| 		alphaOverride = langext.CoalesceOpt(opt.alphaOverride, alphaOverride) | ||||
| 	} | ||||
|  | ||||
| 	if flow { | ||||
| 		y = b.GetY() | ||||
| 	} | ||||
|  | ||||
| 	regName := img.Name | ||||
|  | ||||
| 	var subImageBounds *imageext.PercentageRectangle = nil | ||||
|  | ||||
| 	if imageFit != nil || fillColor != nil || crop != nil { | ||||
|  | ||||
| 		var dataimg image.Image | ||||
| @@ -278,11 +306,14 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | ||||
| 			pxw := w * pdfPixelPerMillimeter | ||||
| 			pxh := h * pdfPixelPerMillimeter | ||||
|  | ||||
| 			dataimg, err = imageext.ObjectFitImage(dataimg, pxw, pxh, *imageFit, fillColor) | ||||
| 			var dataImgRect imageext.PercentageRectangle | ||||
| 			dataimg, dataImgRect, err = imageext.ObjectFitImage(dataimg, pxw, pxh, *imageFit, fillColor) | ||||
| 			if err != nil { | ||||
| 				b.b.SetError(err) | ||||
| 				return | ||||
| 			} | ||||
|  | ||||
| 			subImageBounds = &dataImgRect | ||||
| 		} | ||||
|  | ||||
| 		if dataimg.ColorModel() != color.RGBAModel && dataimg.ColorModel() != color.NRGBAModel { | ||||
| @@ -313,6 +344,12 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | ||||
|  | ||||
| 	} | ||||
|  | ||||
| 	if alphaOverride != nil { | ||||
| 		oldA, oldBMS := b.b.GetAlpha() | ||||
| 		b.b.SetAlpha(alphaOverride.V1, string(alphaOverride.V2)) | ||||
| 		defer func() { b.b.SetAlpha(oldA, oldBMS) }() | ||||
| 	} | ||||
|  | ||||
| 	fpdfOpt := gofpdf.ImageOptions{ | ||||
| 		ImageType:             imageType, | ||||
| 		ReadDpi:               readDpi, | ||||
| @@ -320,4 +357,16 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | ||||
| 	} | ||||
|  | ||||
| 	b.b.ImageOptions(regName, x, y, w, h, flow, fpdfOpt, link, linkStr) | ||||
|  | ||||
| 	if debug { | ||||
| 		b.Rect(w, h, RectOutline, NewPDFRectOpt().X(x).Y(y).LineWidth(0.25).DrawColor(255, 0, 0)) | ||||
|  | ||||
| 		if subImageBounds != nil { | ||||
| 			r := subImageBounds.Of(imageext.Rectangle{X: x, Y: y, W: w, H: h}) | ||||
| 			b.Rect(r.W, r.H, RectOutline, NewPDFRectOpt().X(r.X).Y(r.Y).LineWidth(0.25).DrawColor(255, 0, 0)) | ||||
| 			b.Rect(r.W, r.H, RectFill, NewPDFRectOpt().X(r.X).Y(r.Y).FillColor(255, 0, 0).Alpha(0.2, BlendNormal)) | ||||
| 			b.Line(r.X, r.Y, r.X+r.W, r.Y+r.H, NewPDFLineOpt().LineWidth(0.25).DrawColor(255, 0, 0)) | ||||
| 			b.Line(r.X+r.W, r.Y, r.X, r.Y+r.H, NewPDFLineOpt().LineWidth(0.25).DrawColor(255, 0, 0)) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										96
									
								
								wpdf/wpdfLine.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								wpdf/wpdfLine.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
| package wpdf | ||||
|  | ||||
| import ( | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| ) | ||||
|  | ||||
| type PDFLineOpt struct { | ||||
| 	lineWidth *float64 | ||||
| 	drawColor *PDFColor | ||||
| 	alpha     *dataext.Tuple[float64, PDFBlendMode] | ||||
| 	capStyle  *PDFLineCapStyle | ||||
| 	debug     *bool | ||||
| } | ||||
|  | ||||
| func NewPDFLineOpt() *PDFLineOpt { | ||||
| 	return &PDFLineOpt{} | ||||
| } | ||||
|  | ||||
| func (opt *PDFLineOpt) LineWidth(v float64) *PDFLineOpt { | ||||
| 	opt.lineWidth = &v | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFLineOpt) DrawColor(cr, cg, cb int) *PDFLineOpt { | ||||
| 	opt.drawColor = langext.Ptr(rgbToColor(cr, cg, cb)) | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFLineOpt) DrawColorHex(c uint32) *PDFLineOpt { | ||||
| 	opt.drawColor = langext.Ptr(hexToColor(c)) | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFLineOpt) Alpha(alpha float64, blendMode PDFBlendMode) *PDFLineOpt { | ||||
| 	opt.alpha = &dataext.Tuple[float64, PDFBlendMode]{V1: alpha, V2: blendMode} | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFLineOpt) CapButt() *PDFLineOpt { | ||||
| 	opt.capStyle = langext.Ptr(CapButt) | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFLineOpt) CapSquare() *PDFLineOpt { | ||||
| 	opt.capStyle = langext.Ptr(CapSquare) | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFLineOpt) CapRound() *PDFLineOpt { | ||||
| 	opt.capStyle = langext.Ptr(CapRound) | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFLineOpt) Debug(v bool) *PDFLineOpt { | ||||
| 	opt.debug = &v | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) Line(x1 float64, y1 float64, x2 float64, y2 float64, opts ...*PDFLineOpt) { | ||||
| 	var lineWidth *float64 | ||||
| 	var drawColor *PDFColor | ||||
| 	var alphaOverride *dataext.Tuple[float64, PDFBlendMode] | ||||
| 	capStyle := CapButt | ||||
| 	debug := b.debug | ||||
|  | ||||
| 	for _, opt := range opts { | ||||
| 		lineWidth = langext.CoalesceOpt(opt.lineWidth, lineWidth) | ||||
| 		drawColor = langext.CoalesceOpt(opt.drawColor, drawColor) | ||||
| 		alphaOverride = langext.CoalesceOpt(opt.alpha, alphaOverride) | ||||
| 		capStyle = langext.Coalesce(opt.capStyle, capStyle) | ||||
| 		debug = langext.Coalesce(opt.debug, debug) | ||||
| 	} | ||||
|  | ||||
| 	if lineWidth != nil { | ||||
| 		old := b.GetLineWidth() | ||||
| 		b.SetLineWidth(*lineWidth) | ||||
| 		defer func() { b.SetLineWidth(old) }() | ||||
| 	} | ||||
|  | ||||
| 	if drawColor != nil { | ||||
| 		oldR, oldG, oldB := b.GetDrawColor() | ||||
| 		b.SetDrawColor(drawColor.R, drawColor.G, drawColor.B) | ||||
| 		defer func() { b.SetDrawColor(oldR, oldG, oldB) }() | ||||
| 	} | ||||
|  | ||||
| 	if alphaOverride != nil { | ||||
| 		oldA, oldBMS := b.b.GetAlpha() | ||||
| 		b.b.SetAlpha(alphaOverride.V1, string(alphaOverride.V2)) | ||||
| 		defer func() { b.b.SetAlpha(oldA, oldBMS) }() | ||||
| 	} | ||||
|  | ||||
| 	b.b.SetLineCapStyle(string(capStyle)) | ||||
|  | ||||
| 	b.b.Line(x1, y1, x2, y2) | ||||
| } | ||||
| @@ -1,6 +1,9 @@ | ||||
| package wpdf | ||||
|  | ||||
| import "gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| import ( | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| ) | ||||
|  | ||||
| type PDFMultiCellOpt struct { | ||||
| 	width             *float64 | ||||
| @@ -11,11 +14,13 @@ type PDFMultiCellOpt struct { | ||||
| 	fontNameOverride  *PDFFontFamily | ||||
| 	fontStyleOverride *PDFFontStyle | ||||
| 	fontSizeOverride  *float64 | ||||
| 	alphaOverride     *dataext.Tuple[float64, PDFBlendMode] | ||||
| 	extraLn           *float64 | ||||
| 	x                 *float64 | ||||
| 	textColor         *PDFColor | ||||
| 	borderColor       *PDFColor | ||||
| 	fillColor         *PDFColor | ||||
| 	debug             *bool | ||||
| } | ||||
|  | ||||
| func NewPDFMultiCellOpt() *PDFMultiCellOpt { | ||||
| @@ -119,6 +124,21 @@ func (opt *PDFMultiCellOpt) FillColorHex(c uint32) *PDFMultiCellOpt { | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFMultiCellOpt) Alpha(alpha float64, blendMode PDFBlendMode) *PDFMultiCellOpt { | ||||
| 	opt.alphaOverride = &dataext.Tuple[float64, PDFBlendMode]{V1: alpha, V2: blendMode} | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFMultiCellOpt) Debug(v bool) *PDFMultiCellOpt { | ||||
| 	opt.debug = &v | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFMultiCellOpt) Copy() *PDFMultiCellOpt { | ||||
| 	c := *opt | ||||
| 	return &c | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) MultiCell(txt string, opts ...*PDFMultiCellOpt) { | ||||
|  | ||||
| 	txtTR := b.tr(txt) | ||||
| @@ -131,11 +151,13 @@ func (b *WPDFBuilder) MultiCell(txt string, opts ...*PDFMultiCellOpt) { | ||||
| 	var fontNameOverride *PDFFontFamily | ||||
| 	var fontStyleOverride *PDFFontStyle | ||||
| 	var fontSizeOverride *float64 | ||||
| 	var alphaOverride *dataext.Tuple[float64, PDFBlendMode] | ||||
| 	extraLn := float64(0) | ||||
| 	var x *float64 | ||||
| 	var textColor *PDFColor | ||||
| 	var borderColor *PDFColor | ||||
| 	var fillColor *PDFColor | ||||
| 	debug := b.debug | ||||
|  | ||||
| 	for _, opt := range opts { | ||||
| 		width = langext.Coalesce(opt.width, width) | ||||
| @@ -146,11 +168,13 @@ func (b *WPDFBuilder) MultiCell(txt string, opts ...*PDFMultiCellOpt) { | ||||
| 		fontNameOverride = langext.CoalesceOpt(opt.fontNameOverride, fontNameOverride) | ||||
| 		fontStyleOverride = langext.CoalesceOpt(opt.fontStyleOverride, fontStyleOverride) | ||||
| 		fontSizeOverride = langext.CoalesceOpt(opt.fontSizeOverride, fontSizeOverride) | ||||
| 		alphaOverride = langext.CoalesceOpt(opt.alphaOverride, alphaOverride) | ||||
| 		extraLn = langext.Coalesce(opt.extraLn, extraLn) | ||||
| 		x = langext.CoalesceOpt(opt.x, x) | ||||
| 		textColor = langext.CoalesceOpt(opt.textColor, textColor) | ||||
| 		borderColor = langext.CoalesceOpt(opt.borderColor, borderColor) | ||||
| 		fillColor = langext.CoalesceOpt(opt.fillColor, fillColor) | ||||
| 		debug = langext.Coalesce(opt.debug, debug) | ||||
| 	} | ||||
|  | ||||
| 	if fontNameOverride != nil || fontStyleOverride != nil || fontSizeOverride != nil { | ||||
| @@ -182,12 +206,24 @@ func (b *WPDFBuilder) MultiCell(txt string, opts ...*PDFMultiCellOpt) { | ||||
| 		defer func() { b.SetFillColor(oldColorR, oldColorG, oldColorB) }() | ||||
| 	} | ||||
|  | ||||
| 	if alphaOverride != nil { | ||||
| 		oldA, oldBMS := b.b.GetAlpha() | ||||
| 		b.b.SetAlpha(alphaOverride.V1, string(alphaOverride.V2)) | ||||
| 		defer func() { b.b.SetAlpha(oldA, oldBMS) }() | ||||
| 	} | ||||
|  | ||||
| 	if x != nil { | ||||
| 		b.b.SetX(*x) | ||||
| 	} | ||||
|  | ||||
| 	xBefore, yBefore := b.b.GetXY() | ||||
|  | ||||
| 	b.b.MultiCell(width, height, txtTR, string(border), string(align), fill) | ||||
|  | ||||
| 	if debug { | ||||
| 		b.Rect(b.GetPageWidth()-xBefore-b.GetMarginRight(), b.GetY()-yBefore, RectOutline, NewPDFRectOpt().X(xBefore).Y(yBefore).LineWidth(0.25).DrawColor(0, 128, 0)) | ||||
| 	} | ||||
|  | ||||
| 	if extraLn != 0 { | ||||
| 		b.b.Ln(extraLn) | ||||
| 	} | ||||
|   | ||||
| @@ -1,6 +1,9 @@ | ||||
| package wpdf | ||||
|  | ||||
| import "gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| import ( | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| ) | ||||
|  | ||||
| type PDFRectOpt struct { | ||||
| 	x         *float64 | ||||
| @@ -8,10 +11,12 @@ type PDFRectOpt struct { | ||||
| 	lineWidth *float64 | ||||
| 	drawColor *PDFColor | ||||
| 	fillColor *PDFColor | ||||
| 	alpha     *dataext.Tuple[float64, PDFBlendMode] | ||||
| 	radiusTL  *float64 | ||||
| 	radiusTR  *float64 | ||||
| 	radiusBR  *float64 | ||||
| 	radiusBL  *float64 | ||||
| 	debug     *bool | ||||
| } | ||||
|  | ||||
| func NewPDFRectOpt() *PDFRectOpt { | ||||
| @@ -81,16 +86,28 @@ func (opt *PDFRectOpt) RadiusBR(radius float64) *PDFRectOpt { | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFRectOpt) Alpha(alpha float64, blendMode PDFBlendMode) *PDFRectOpt { | ||||
| 	opt.alpha = &dataext.Tuple[float64, PDFBlendMode]{V1: alpha, V2: blendMode} | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (opt *PDFRectOpt) Debug(v bool) *PDFRectOpt { | ||||
| 	opt.debug = &v | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) Rect(w float64, h float64, styleStr PDFRectStyle, opts ...*PDFRectOpt) { | ||||
| 	x := b.GetX() | ||||
| 	y := b.GetY() | ||||
| 	var lineWidth *float64 | ||||
| 	var drawColor *PDFColor | ||||
| 	var fillColor *PDFColor | ||||
| 	var alphaOverride *dataext.Tuple[float64, PDFBlendMode] | ||||
| 	radiusTL := float64(0) | ||||
| 	radiusTR := float64(0) | ||||
| 	radiusBR := float64(0) | ||||
| 	radiusBL := float64(0) | ||||
| 	debug := b.debug | ||||
|  | ||||
| 	for _, opt := range opts { | ||||
| 		x = langext.Coalesce(opt.x, x) | ||||
| @@ -98,10 +115,12 @@ func (b *WPDFBuilder) Rect(w float64, h float64, styleStr PDFRectStyle, opts ... | ||||
| 		lineWidth = langext.CoalesceOpt(opt.lineWidth, lineWidth) | ||||
| 		drawColor = langext.CoalesceOpt(opt.drawColor, drawColor) | ||||
| 		fillColor = langext.CoalesceOpt(opt.fillColor, fillColor) | ||||
| 		alphaOverride = langext.CoalesceOpt(opt.alpha, alphaOverride) | ||||
| 		radiusTL = langext.Coalesce(opt.radiusTL, radiusTL) | ||||
| 		radiusTR = langext.Coalesce(opt.radiusTR, radiusTR) | ||||
| 		radiusBR = langext.Coalesce(opt.radiusBR, radiusBR) | ||||
| 		radiusBL = langext.Coalesce(opt.radiusBL, radiusBL) | ||||
| 		debug = langext.Coalesce(opt.debug, debug) | ||||
| 	} | ||||
|  | ||||
| 	if lineWidth != nil { | ||||
| @@ -122,5 +141,11 @@ func (b *WPDFBuilder) Rect(w float64, h float64, styleStr PDFRectStyle, opts ... | ||||
| 		defer func() { b.SetFillColor(oldR, oldG, oldB) }() | ||||
| 	} | ||||
|  | ||||
| 	if alphaOverride != nil { | ||||
| 		oldA, oldBMS := b.b.GetAlpha() | ||||
| 		b.b.SetAlpha(alphaOverride.V1, string(alphaOverride.V2)) | ||||
| 		defer func() { b.b.SetAlpha(oldA, oldBMS) }() | ||||
| 	} | ||||
|  | ||||
| 	b.b.RoundedRectExt(x, y, w, h, radiusTL, radiusTR, radiusBR, radiusBL, string(styleStr)) | ||||
| } | ||||
|   | ||||
							
								
								
									
										346
									
								
								wpdf/wpdfTable.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										346
									
								
								wpdf/wpdfTable.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,346 @@ | ||||
| package wpdf | ||||
|  | ||||
| import ( | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/exerr" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/rext" | ||||
| 	"regexp" | ||||
| 	"strconv" | ||||
| ) | ||||
|  | ||||
| // Column specifier: | ||||
| // | ||||
| // - `{number}`:             Use this amount of space | ||||
| // - `auto`:                 Use the needed space for the content | ||||
| // - `*` / `fr`:             Use the remaining space, evenly distributed, shrink down to auto | ||||
| // - `{num}fr` / `{num}*`:   Use the remaining space, evenly distributed (weighted), shrink down to auto | ||||
| // | ||||
| // # TableBuilder | ||||
| //    - PadX/PadY:            Padding between cells | ||||
| //    - DefaultStyle:         Default style for cells | ||||
| // | ||||
| // # TableCellStyleOpt | ||||
| //    - MultiCell:            Use wpdf.MultiCell() instead of wpdf.Cell()  --> supports linebreaks | ||||
| //    - Ellipsize:            Ellipsize text if too long | ||||
| //    - PaddingHorz:          Additional horizontal padding inside of cell to space text around | ||||
| //    - PDFCellOpt:           Normal styling options (evtl not all are supported, depending on MultiCell: true/false) | ||||
|  | ||||
| var regexTableColumnSpecFr = rext.W(regexp.MustCompile(`^(?P<num>[0-9]*)(fr|\*)$`)) | ||||
|  | ||||
| type TableBuilder struct { | ||||
| 	builder *WPDFBuilder | ||||
|  | ||||
| 	padx             float64 | ||||
| 	pady             float64 | ||||
| 	rows             []tableRow | ||||
| 	defaultCellStyle *TableCellStyleOpt | ||||
| 	columnWidths     *[]string | ||||
| 	debug            *bool | ||||
| } | ||||
|  | ||||
| type TableCell struct { | ||||
| 	Content string | ||||
| 	Style   TableCellStyleOpt | ||||
| } | ||||
|  | ||||
| type TableCellStyleOpt struct { | ||||
| 	MultiCell   *bool | ||||
| 	Ellipsize   *bool | ||||
| 	PaddingHorz *float64 | ||||
| 	MinWidth    *float64 | ||||
|  | ||||
| 	PDFCellOpt | ||||
| } | ||||
|  | ||||
| type tableRow struct { | ||||
| 	cells []TableCell | ||||
| } | ||||
|  | ||||
| func (r tableRow) maxFontSize(defaultFontSize float64) float64 { | ||||
| 	mfs := defaultFontSize | ||||
| 	for _, cell := range r.cells { | ||||
| 		if cell.Style.fontSizeOverride != nil { | ||||
| 			mfs = max(mfs, *cell.Style.fontSizeOverride) | ||||
| 		} | ||||
| 	} | ||||
| 	return mfs | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) Widths(v ...string) *TableBuilder { | ||||
| 	b.columnWidths = &v | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) DefaultStyle(s TableCellStyleOpt) *TableBuilder { | ||||
| 	b.defaultCellStyle = &s | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) PadX(v float64) *TableBuilder { | ||||
| 	b.padx = v | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) PadY(v float64) *TableBuilder { | ||||
| 	b.pady = v | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) AddRow(cells ...TableCell) *TableBuilder { | ||||
| 	b.rows = append(b.rows, tableRow{cells: cells}) | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) AddRowWithStyle(style TableCellStyleOpt, cells ...string) *TableBuilder { | ||||
| 	tcels := make([]TableCell, 0, len(cells)) | ||||
| 	for _, cell := range cells { | ||||
| 		tcels = append(tcels, TableCell{Content: cell, Style: style}) | ||||
| 	} | ||||
|  | ||||
| 	b.rows = append(b.rows, tableRow{cells: tcels}) | ||||
|  | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) AddRowDefaultStyle(cells ...string) *TableBuilder { | ||||
| 	tcels := make([]TableCell, 0, len(cells)) | ||||
| 	for _, cell := range cells { | ||||
| 		tcels = append(tcels, TableCell{Content: cell, Style: langext.Coalesce(b.defaultCellStyle, TableCellStyleOpt{})}) | ||||
| 	} | ||||
|  | ||||
| 	b.rows = append(b.rows, tableRow{cells: tcels}) | ||||
|  | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) Build() { | ||||
| 	builder := b.builder | ||||
|  | ||||
| 	debug := langext.Coalesce(b.debug, b.builder.debug) | ||||
|  | ||||
| 	if len(b.rows) == 0 { | ||||
| 		return // nothing to do | ||||
| 	} | ||||
|  | ||||
| 	_, pageHeight := builder.FPDF().GetPageSize() | ||||
| 	pbEnabled, pbMargin := builder.FPDF().GetAutoPageBreak() | ||||
|  | ||||
| 	builder.FPDF().SetAutoPageBreak(false, 0) // manually handle pagebreak in tables | ||||
| 	defer func() { builder.FPDF().SetAutoPageBreak(pbEnabled, pbMargin) }() | ||||
|  | ||||
| 	columnWidths := b.calculateColumns() | ||||
|  | ||||
| 	columnCount := len(columnWidths) | ||||
|  | ||||
| 	for i, dat := range b.rows { | ||||
| 		if len(dat.cells) != columnCount { | ||||
| 			builder.FPDF().SetError(exerr.New(exerr.TypeInternal, "data must have the same length as header").Int("idx", i).Build()) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	defaultFontSize, _ := builder.FPDF().GetFontSize() | ||||
|  | ||||
| 	for rowIdx, row := range b.rows { | ||||
| 		nextY := builder.GetY() | ||||
| 		for cellIdx, cell := range row.cells { | ||||
|  | ||||
| 			str := cell.Content | ||||
| 			style := cell.Style | ||||
|  | ||||
| 			ellipsize := langext.Coalesce(style.Ellipsize, true) | ||||
| 			cellPaddingHorz := langext.Coalesce(style.PaddingHorz, 2) | ||||
|  | ||||
| 			bx := builder.GetX() | ||||
| 			by := builder.GetY() | ||||
|  | ||||
| 			cellWidth := columnWidths[cellIdx] | ||||
|  | ||||
| 			if langext.Coalesce(style.MultiCell, true) { | ||||
|  | ||||
| 				builder.MultiCell(str, style.PDFCellOpt.Copy().ToMulti().Width(cellWidth).Debug(debug)) | ||||
|  | ||||
| 			} else { | ||||
|  | ||||
| 				if ellipsize { | ||||
| 					if builder.GetStringWidth(str, style.PDFCellOpt) > (cellWidth - cellPaddingHorz) { | ||||
| 						for builder.GetStringWidth(str+"...", style.PDFCellOpt) > (cellWidth-cellPaddingHorz) && len(str) > 0 { | ||||
| 							str = str[:len(str)-1] | ||||
| 						} | ||||
| 						str += "..." | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				builder.Cell(str, style.PDFCellOpt.Copy().Width(cellWidth).Debug(debug)) | ||||
|  | ||||
| 			} | ||||
|  | ||||
| 			nextY = max(nextY, builder.GetY()) | ||||
| 			builder.SetXY(bx+cellWidth+b.padx, by) | ||||
| 		} | ||||
| 		builder.SetY(nextY + b.pady) | ||||
|  | ||||
| 		if rowIdx < len(b.rows)-1 && pbEnabled && (builder.GetY()+b.rows[rowIdx+1].maxFontSize(defaultFontSize)) > (pageHeight-pbMargin) { | ||||
| 			builder.FPDF().AddPage() | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) calculateColumns() []float64 { | ||||
| 	pageWidthTotal, _ := b.builder.FPDF().GetPageSize() | ||||
| 	marginLeft, _, marginRight, _ := b.builder.FPDF().GetMargins() | ||||
| 	pageWidth := pageWidthTotal - marginLeft - marginRight | ||||
|  | ||||
| 	columnDef := make([]string, 0) | ||||
|  | ||||
| 	if b.columnWidths != nil { | ||||
| 		columnDef = *b.columnWidths | ||||
| 	} else if len(b.rows) > 0 { | ||||
| 		columnDef = make([]string, len(b.rows[0].cells)) | ||||
| 		for i := range columnDef { | ||||
| 			columnDef[i] = "*" | ||||
| 		} | ||||
| 	} else { | ||||
| 		return []float64{} | ||||
| 	} | ||||
|  | ||||
| 	columnWidths := make([]float64, len(columnDef)) | ||||
|  | ||||
| 	frColumnWidthCount := 0 | ||||
| 	frColumnWeights := make([]float64, len(columnDef)) | ||||
| 	remainingWidth := pageWidth - (float64(len(columnDef)-1) * b.padx) | ||||
| 	autoWidths := make([]float64, len(columnDef)) | ||||
|  | ||||
| 	for colIdx := range columnDef { | ||||
| 		w := float64(0) | ||||
| 		for _, row := range b.rows { | ||||
| 			if len(row.cells) > colIdx { | ||||
| 				w = max(w, b.builder.GetStringWidth(row.cells[colIdx].Content, row.cells[colIdx].Style.PDFCellOpt)) | ||||
| 			} | ||||
| 		} | ||||
| 		autoWidths[colIdx] = w | ||||
| 	} | ||||
|  | ||||
| 	for colIdx, col := range columnDef { | ||||
|  | ||||
| 		maxPadHorz := float64(0) | ||||
|  | ||||
| 		minWidth := float64(0) | ||||
| 		for _, row := range b.rows { | ||||
| 			if len(row.cells) > colIdx { | ||||
|  | ||||
| 				ph := langext.Coalesce(row.cells[colIdx].Style.PaddingHorz, 2) | ||||
| 				mw := langext.Coalesce(row.cells[colIdx].Style.MinWidth, 0) | ||||
|  | ||||
| 				minWidth = max(minWidth, ph+mw) | ||||
|  | ||||
| 				maxPadHorz = max(maxPadHorz, ph) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if col == "auto" { | ||||
|  | ||||
| 			w := max(autoWidths[colIdx]+maxPadHorz, minWidth) | ||||
|  | ||||
| 			columnWidths[colIdx] = w | ||||
| 			remainingWidth -= w | ||||
|  | ||||
| 		} else if match, ok := regexTableColumnSpecFr.MatchFirst(col); ok { | ||||
|  | ||||
| 			if match.GroupByName("num").Value() == "" { | ||||
| 				w := minWidth | ||||
|  | ||||
| 				frColumnWidthCount += 1 | ||||
| 				frColumnWeights[colIdx] = 1 | ||||
| 				columnWidths[colIdx] = w | ||||
| 				remainingWidth -= w | ||||
| 			} else { | ||||
| 				w := minWidth | ||||
|  | ||||
| 				n, _ := strconv.Atoi(match.GroupByName("num").Value()) | ||||
| 				frColumnWidthCount += n | ||||
| 				frColumnWeights[colIdx] = float64(n) | ||||
| 				columnWidths[colIdx] = w | ||||
| 				remainingWidth -= w | ||||
| 			} | ||||
|  | ||||
| 		} else { | ||||
|  | ||||
| 			if w, err := strconv.ParseFloat(col, 64); err == nil { | ||||
| 				w = max(w, minWidth) | ||||
|  | ||||
| 				columnWidths[colIdx] = w | ||||
| 				remainingWidth -= w | ||||
| 			} else { | ||||
| 				b.builder.FPDF().SetError(exerr.New(exerr.TypeInternal, "invalid column width").Str("width", col).Build()) | ||||
| 				w = max(w, minWidth) | ||||
|  | ||||
| 				columnWidths[colIdx] = w | ||||
| 				remainingWidth -= w | ||||
| 				return nil | ||||
| 			} | ||||
|  | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if remainingWidth < 0 { | ||||
| 		// no remaining space to distribute | ||||
| 		return columnWidths | ||||
| 	} | ||||
|  | ||||
| 	for i, _ := range columnDef { | ||||
| 		if frColumnWeights[i] != 0 { | ||||
| 			w := min(autoWidths[i], (remainingWidth/float64(frColumnWidthCount))*frColumnWeights[i]) | ||||
| 			remainingWidth += columnWidths[i] | ||||
| 			columnWidths[i] = w | ||||
| 			remainingWidth -= w | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if remainingWidth > 0 { | ||||
| 		for i, _ := range columnDef { | ||||
| 			if frColumnWeights[i] != 0 { | ||||
| 				addW := (remainingWidth / float64(frColumnWidthCount)) * frColumnWeights[i] | ||||
| 				columnWidths[i] += addW | ||||
| 				remainingWidth -= addW | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return columnWidths | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) RowCount() int { | ||||
| 	return len(b.rows) | ||||
| } | ||||
|  | ||||
| func (b *TableBuilder) Debug(v bool) *TableBuilder { | ||||
| 	b.debug = &v | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| func (b *WPDFBuilder) Table() *TableBuilder { | ||||
| 	return &TableBuilder{ | ||||
| 		builder:          b, | ||||
| 		rows:             make([]tableRow, 0), | ||||
| 		pady:             2, | ||||
| 		padx:             2, | ||||
| 		defaultCellStyle: defaultTableStyle(), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func defaultTableStyle() *TableCellStyleOpt { | ||||
| 	return &TableCellStyleOpt{ | ||||
| 		PDFCellOpt: *NewPDFCellOpt(). | ||||
| 			FontSize(float64(8)). | ||||
| 			Border(BorderFull). | ||||
| 			BorderColorHex(uint32(0x666666)). | ||||
| 			FillColorHex(uint32(0xF0F0F0)). | ||||
| 			TextColorHex(uint32(0x000000)). | ||||
| 			FillBackground(true), | ||||
| 		MinWidth:  langext.Ptr(float64(5)), | ||||
| 		Ellipsize: langext.PTrue, | ||||
| 		MultiCell: langext.PFalse, | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										88
									
								
								wpdf/wpdf_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								wpdf/wpdf_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | ||||
| package wpdf | ||||
|  | ||||
| import ( | ||||
| 	_ "embed" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/imageext" | ||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||
| 	"os" | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| //go:embed logo.png | ||||
| var logoData []byte | ||||
|  | ||||
| func TestPDFBuilder(t *testing.T) { | ||||
| 	builder := NewPDFBuilder(Portrait, SizeA4, true) | ||||
|  | ||||
| 	builder.Debug(true) | ||||
|  | ||||
| 	logoRef := builder.RegisterImage(logoData) | ||||
|  | ||||
| 	builder.SetMargins(PDFMargins{Left: 15, Top: 40, Right: 10}) | ||||
| 	builder.AddPage() | ||||
|  | ||||
| 	builder.SetFont(FontHelvetica, Normal, 10) | ||||
| 	builder.Cell("Neueinrichtung deiner Entgeltumwandlung", NewPDFCellOpt().Bold().FontSize(20)) | ||||
| 	builder.Ln(10) | ||||
|  | ||||
| 	builder.SetFont(FontHelvetica, Normal, 10) | ||||
| 	builder.Cell("Hello World", NewPDFCellOpt().Width(50).Align(AlignHorzCenter).LnPos(BreakToRight)) | ||||
| 	builder.IncX(10) | ||||
| 	builder.Cell("Second Text", NewPDFCellOpt().AutoWidth().AutoWidthPaddingX(2).LnPos(BreakToRight)) | ||||
| 	builder.Ln(10) | ||||
|  | ||||
| 	builder.MultiCell("Im Fall einer individuellen Entgeltumwandlung ist die Zuschussverpflichtung auf der Grundlage des Betriebsrentenstärkungsgesetzes in der gesetzlich vorgeschriebenen Höhe (§ 1a Abs. 1a BetrAVG), über den arbeitgeberfinanzierten Zuschuss erfüllt.") | ||||
| 	builder.Ln(4) | ||||
|  | ||||
| 	builder.Image(logoRef, NewPDFImageOpt().X(90).Y(160).Width(70).Height(30).ImageFit(imageext.ImageFitContainCenter)) | ||||
|  | ||||
| 	builder.Ln(4) | ||||
|  | ||||
| 	cellStyleHeader := TableCellStyleOpt{ | ||||
| 		PDFCellOpt: *NewPDFCellOpt(). | ||||
| 			FontSize(float64(8)). | ||||
| 			BorderColorHex(uint32(0x666666)). | ||||
| 			Border(BorderFull). | ||||
| 			FillColorHex(uint32(0xC0C0C0)). | ||||
| 			FillBackground(true). | ||||
| 			TextColorHex(uint32(0x000000)). | ||||
| 			Align(AlignHorzCenter). | ||||
| 			Bold(), | ||||
| 		MinWidth:  langext.Ptr(float64(5)), | ||||
| 		Ellipsize: langext.PTrue, | ||||
| 		MultiCell: langext.PFalse, | ||||
| 	} | ||||
|  | ||||
| 	cellStyleMulti := TableCellStyleOpt{ | ||||
| 		PDFCellOpt: *NewPDFCellOpt(). | ||||
| 			FontSize(float64(8)). | ||||
| 			BorderColorHex(uint32(0x666666)). | ||||
| 			Border(BorderFull). | ||||
| 			FillColorHex(uint32(0xC060C0)). | ||||
| 			FillBackground(true). | ||||
| 			TextColorHex(uint32(0x000000)), | ||||
| 		MinWidth:  langext.Ptr(float64(5)), | ||||
| 		Ellipsize: langext.PFalse, | ||||
| 		MultiCell: langext.PTrue, | ||||
| 	} | ||||
|  | ||||
| 	builder.Table(). | ||||
| 		Widths("auto", "20", "1fr", "20"). | ||||
| 		PadX(2). | ||||
| 		PadY(2). | ||||
| 		AddRowWithStyle(cellStyleHeader, "test", "hello", "123", "end"). | ||||
| 		AddRowDefaultStyle("test", "hello", "123", "end"). | ||||
| 		AddRowDefaultStyle("123", "helasdsalo", "a", "enwqad"). | ||||
| 		AddRowDefaultStyle("123asd", "TrimMeTrimMeTrimMeTrimMe", "a", "enwqad"). | ||||
| 		AddRowWithStyle(cellStyleMulti, "123", "helasdsalo", "a", "MultiCell: enwqad enw\nqad enwqad enwqad enwqad enwqad enwqad enwqad enwqad"). | ||||
| 		AddRowDefaultStyle("123", "helasdsalo", "a", "enwqad"). | ||||
| 		Debug(false). | ||||
| 		Build() | ||||
|  | ||||
| 	bin, err := builder.Build() | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
|  | ||||
| 	_ = os.WriteFile("wpdf_test.pdf", bin, 0644) | ||||
| } | ||||
		Reference in New Issue
	
	Block a user