Compare commits
	
		
			8 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1e98d351ce | |||
| c40bdc8e9e | |||
| 7204562879 | |||
| 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 { | func (j jsonHTTPResponse) jsonRenderer(g *gin.Context) json.GoJsonRender { | ||||||
| 	var f *string | 	var f *string | ||||||
| 	if jsonfilter := g.GetString("goext.jsonfilter"); jsonfilter != "" { | 	if jsonfilter := g.GetString(jsonFilterKey); jsonfilter != "" { | ||||||
| 		f = &jsonfilter | 		f = &jsonfilter | ||||||
| 	} | 	} | ||||||
| 	return json.GoJsonRender{Data: j.data, NilSafeSlices: true, NilSafeMaps: true, Filter: f} | 	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 { | 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 { | 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 { | 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) { | 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/xid v1.5.0 | ||||||
| 	github.com/rs/zerolog v1.33.0 | 	github.com/rs/zerolog v1.33.0 | ||||||
| 	go.mongodb.org/mongo-driver v1.16.0 | 	go.mongodb.org/mongo-driver v1.16.0 | ||||||
| 	golang.org/x/crypto v0.25.0 | 	golang.org/x/crypto v0.26.0 | ||||||
| 	golang.org/x/sys v0.22.0 | 	golang.org/x/sys v0.23.0 | ||||||
| 	golang.org/x/term v0.22.0 | 	golang.org/x/term v0.23.0 | ||||||
| ) | ) | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/disintegration/imaging v1.6.2 | 	github.com/disintegration/imaging v1.6.2 | ||||||
| 	github.com/jung-kurt/gofpdf v1.16.2 | 	github.com/jung-kurt/gofpdf v1.16.2 | ||||||
| 	golang.org/x/sync v0.7.0 | 	golang.org/x/sync v0.8.0 | ||||||
| ) | ) | ||||||
|  |  | ||||||
| require ( | 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/bytedance/sonic/loader v0.2.0 // indirect | ||||||
| 	github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect | 	github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect | ||||||
| 	github.com/chenzhuoyu/iasm v0.9.1 // 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/scram v1.1.2 // indirect | ||||||
| 	github.com/xdg-go/stringprep v1.0.4 // indirect | 	github.com/xdg-go/stringprep v1.0.4 // indirect | ||||||
| 	github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78 // indirect | 	github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78 // indirect | ||||||
| 	golang.org/x/arch v0.8.0 // indirect | 	golang.org/x/arch v0.9.0 // indirect | ||||||
| 	golang.org/x/image v0.18.0 // indirect | 	golang.org/x/image v0.19.0 // indirect | ||||||
| 	golang.org/x/net v0.27.0 // indirect | 	golang.org/x/net v0.28.0 // indirect | ||||||
| 	golang.org/x/text v0.16.0 // indirect | 	golang.org/x/text v0.17.0 // indirect | ||||||
| 	google.golang.org/protobuf v1.34.2 // indirect | 	google.golang.org/protobuf v1.34.2 // indirect | ||||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||||
| 	modernc.org/libc v1.37.6 // 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.11.9/go.mod h1:LysEHSvpvDySVdC2f87zGWf6CIKJcAvqab1ZaiQtds4= | ||||||
| github.com/bytedance/sonic v1.12.0 h1:YGPgxF9xzaCNvd/ZKdQ28yRovhfMFZQjuk6fKBzZ3ls= | 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.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.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 h1:c+e5Pt1k/cy5wMveRDyk2X4B9hF4g7an8N3zCYjJFNM= | ||||||
| github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU= | 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.7.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys= | ||||||
| golang.org/x/arch v0.8.0 h1:3wRIsP3pM4yUptoR96otTUOXI367OS0+c9eeRi9doIc= | 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.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-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-20200302210943-78000ba7a073/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||||
| golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | 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.24.0/go.mod h1:Z1PMYSOR5nyMcyAVAIQSKCDwalqy85Aqn1x3Ws4L5DM= | ||||||
| golang.org/x/crypto v0.25.0 h1:ypSNr+bnYL2YhwoMt2zPxHFmbAN1KZs/njMG3hxUp30= | 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.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-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 h1:hVwzHzIUGRjiF7EcUjqNxk3NCfkPxbDKRdnNE1Rpg0U= | ||||||
| golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | 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.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 h1:jGzIakQa/ZXI1I0Fxvaa9W7yP25TqT6cHIHn+6CqvSQ= | ||||||
| golang.org/x/image v0.18.0/go.mod h1:4yyo5vMFQjVjUcVk4jEQcU9MGy/rulF5WvUILseCM2E= | 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/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-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | 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.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 h1:5K3Njcw06/l2y9vpGCSdcxWOYHOUk3dVNGDXN+FvAys= | ||||||
| golang.org/x/net v0.27.0/go.mod h1:dDi0PyhWNoiUOrAS8uXv/vnScO4wnHQO4mj9fn/RytE= | 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-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.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 h1:5BMeUDZ7vkXGfEr1x9B4bRcTH4lpkTkpdh0T/J+qjbQ= | ||||||
| golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | 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 h1:YsImfSBoP9QPYL0xyKJPq0gcaJdG3rInoqxTWbfQu9M= | ||||||
| golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | 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-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-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/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.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 h1:RI27ohtqKCnwULzJLqkv897zojh5/DwS/ENaMzUOaWI= | ||||||
| golang.org/x/sys v0.22.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | 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-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.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||||
| golang.org/x/term v0.16.0 h1:m+B6fahuftsE9qjo0VWp2FW0mB3MTJvR0BaMQrq0pmE= | 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.21.0/go.mod h1:ooXLefLobQVslOqselCNF4SxFAaoS6KujMbsGzSDmX0= | ||||||
| golang.org/x/term v0.22.0 h1:BbsgPEJULsl2fV/AT3v15Mjva5yXKQDyKf+TbDz7QJk= | 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.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.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.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
| golang.org/x/text v0.3.6/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.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= | ||||||
| golang.org/x/text v0.16.0 h1:a94ExnEXNtEwYLGJSIUxnWoxoRz/ZcCsV63ROupILh4= | 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.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-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.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||||
| golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= | golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| package goext | package goext | ||||||
|  |  | ||||||
| const GoextVersion = "0.0.491" | const GoextVersion = "0.0.499" | ||||||
|  |  | ||||||
| const GoextVersionTimestamp = "2024-07-31T00:15:09+0200" | const GoextVersionTimestamp = "2024-08-07T18:34:22+0200" | ||||||
|   | |||||||
| @@ -788,7 +788,7 @@ FieldLoop: | |||||||
|  |  | ||||||
| 		if f.omitEmpty && isEmptyValue(fv) { | 		if f.omitEmpty && isEmptyValue(fv) { | ||||||
| 			continue | 			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 | 			continue | ||||||
| 		} | 		} | ||||||
| 		e.WriteByte(next) | 		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 { | func newStructEncoder(t reflect.Type, tagkey string) encoderFunc { | ||||||
| 	se := structEncoder{fields: cachedTypeFields(t, tagkey)} | 	se := structEncoder{fields: cachedTypeFields(t, tagkey)} | ||||||
| 	return se.encode | 	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 | 	iw := img.Bounds().Size().X | ||||||
| 	ih := img.Bounds().Size().Y | 	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(), &image.Uniform{C: fillColor}, image.Pt(0, 0), draw.Src) | ||||||
| 		draw.Draw(newImg, newImg.Bounds(), img, image.Pt(0, 0), draw.Over) | 		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 { | 	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, | 		// 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 | 		// 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, newImg.Bounds(), &image.Uniform{C: fillColor}, image.Pt(0, 0), draw.Src) | ||||||
| 		draw.Draw(newImg, destBounds, img, image.Pt(0, 0), draw.Over) | 		draw.Draw(newImg, destBounds, img, image.Pt(0, 0), draw.Over) | ||||||
|  |  | ||||||
| 		return newImg, nil | 		return newImg, calcRelativeRect(destBounds, newImg.Bounds()), nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if fit == ImageFitStretch { | 	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(), &image.Uniform{C: fillColor}, image.Pt(0, 0), draw.Src) | ||||||
| 		draw.Draw(newImg, newImg.Bounds(), img, image.Pt(0, 0), draw.Over) | 		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) { | 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 | 	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 { | 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)) | 	r := make([]TR, 0, len(inmap)) | ||||||
| 	for k, v := range 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 | 
							
								
								
									
										85
									
								
								wpdf/wpdf.go
									
									
									
									
									
								
							
							
						
						
									
										85
									
								
								wpdf/wpdf.go
									
									
									
									
									
								
							| @@ -3,6 +3,7 @@ package wpdf | |||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
| 	"github.com/jung-kurt/gofpdf" | 	"github.com/jung-kurt/gofpdf" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type WPDFBuilder struct { | type WPDFBuilder struct { | ||||||
| @@ -13,6 +14,7 @@ type WPDFBuilder struct { | |||||||
| 	fontName    PDFFontFamily | 	fontName    PDFFontFamily | ||||||
| 	fontStyle   PDFFontStyle | 	fontStyle   PDFFontStyle | ||||||
| 	fontSize    float64 | 	fontSize    float64 | ||||||
|  | 	debug       bool | ||||||
| } | } | ||||||
|  |  | ||||||
| type PDFMargins struct { | type PDFMargins struct { | ||||||
| @@ -61,6 +63,19 @@ func (b *WPDFBuilder) SetMargins(v PDFMargins) { | |||||||
|  |  | ||||||
| func (b *WPDFBuilder) AddPage() { | func (b *WPDFBuilder) AddPage() { | ||||||
| 	b.b.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) { | 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) | 	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) { | func (b *WPDFBuilder) SetCellSpacing(h float64) { | ||||||
| 	b.cellSpacing = h | 	b.cellSpacing = h | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *WPDFBuilder) Ln(h float64) { | func (b *WPDFBuilder) Ln(h float64) { | ||||||
|  | 	xBefore, yBefore := b.GetXY() | ||||||
|  |  | ||||||
| 	b.b.Ln(h) | 	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) { | func (b *WPDFBuilder) Build() ([]byte, error) { | ||||||
| @@ -192,6 +233,48 @@ func (b *WPDFBuilder) GetWorkAreaWidth() float64 { | |||||||
| 	return b.GetPageWidth() - b.GetMarginLeft() - b.GetMarginRight() | 	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) Bookmark(txtStr string, level int, y float64) { | ||||||
|  | 	b.b.Bookmark(b.tr(txtStr), level, y) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | 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) | 	return b.b.GetStringWidth(str) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (b *WPDFBuilder) Debug(v bool) { | ||||||
|  | 	b.debug = v | ||||||
|  | } | ||||||
|   | |||||||
| @@ -1,6 +1,9 @@ | |||||||
| package wpdf | 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 { | type PDFCellOpt struct { | ||||||
| 	width             *float64 | 	width             *float64 | ||||||
| @@ -14,6 +17,7 @@ type PDFCellOpt struct { | |||||||
| 	fontNameOverride  *PDFFontFamily | 	fontNameOverride  *PDFFontFamily | ||||||
| 	fontStyleOverride *PDFFontStyle | 	fontStyleOverride *PDFFontStyle | ||||||
| 	fontSizeOverride  *float64 | 	fontSizeOverride  *float64 | ||||||
|  | 	alphaOverride     *dataext.Tuple[float64, PDFBlendMode] | ||||||
| 	extraLn           *float64 | 	extraLn           *float64 | ||||||
| 	x                 *float64 | 	x                 *float64 | ||||||
| 	autoWidth         *bool | 	autoWidth         *bool | ||||||
| @@ -21,6 +25,7 @@ type PDFCellOpt struct { | |||||||
| 	borderColor       *PDFColor | 	borderColor       *PDFColor | ||||||
| 	fillColor         *PDFColor | 	fillColor         *PDFColor | ||||||
| 	autoWidthPaddingX *float64 | 	autoWidthPaddingX *float64 | ||||||
|  | 	debug             *bool | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewPDFCellOpt() *PDFCellOpt { | func NewPDFCellOpt() *PDFCellOpt { | ||||||
| @@ -149,12 +154,45 @@ func (opt *PDFCellOpt) FillColorHex(c uint32) *PDFCellOpt { | |||||||
| 	return opt | 	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) { | func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | ||||||
|  |  | ||||||
| 	txtTR := b.tr(txt) | 	txtTR := b.tr(txt) | ||||||
|  |  | ||||||
| 	width := float64(0) | 	width := float64(0) | ||||||
| 	height := b.cellHeight + b.cellSpacing | 	var height *float64 = nil | ||||||
| 	border := BorderNone | 	border := BorderNone | ||||||
| 	ln := BreakToNextLine | 	ln := BreakToNextLine | ||||||
| 	align := AlignLeft | 	align := AlignLeft | ||||||
| @@ -164,6 +202,7 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | |||||||
| 	var fontNameOverride *PDFFontFamily | 	var fontNameOverride *PDFFontFamily | ||||||
| 	var fontStyleOverride *PDFFontStyle | 	var fontStyleOverride *PDFFontStyle | ||||||
| 	var fontSizeOverride *float64 | 	var fontSizeOverride *float64 | ||||||
|  | 	var alphaOverride *dataext.Tuple[float64, PDFBlendMode] | ||||||
| 	extraLn := float64(0) | 	extraLn := float64(0) | ||||||
| 	var x *float64 | 	var x *float64 | ||||||
| 	autoWidth := false | 	autoWidth := false | ||||||
| @@ -171,10 +210,11 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | |||||||
| 	var borderColor *PDFColor | 	var borderColor *PDFColor | ||||||
| 	var fillColor *PDFColor | 	var fillColor *PDFColor | ||||||
| 	autoWidthPaddingX := float64(0) | 	autoWidthPaddingX := float64(0) | ||||||
|  | 	debug := b.debug | ||||||
|  |  | ||||||
| 	for _, opt := range opts { | 	for _, opt := range opts { | ||||||
| 		width = langext.Coalesce(opt.width, width) | 		width = langext.Coalesce(opt.width, width) | ||||||
| 		height = langext.Coalesce(opt.height, height) | 		height = langext.CoalesceOpt(opt.height, height) | ||||||
| 		border = langext.Coalesce(opt.border, border) | 		border = langext.Coalesce(opt.border, border) | ||||||
| 		ln = langext.Coalesce(opt.ln, ln) | 		ln = langext.Coalesce(opt.ln, ln) | ||||||
| 		align = langext.Coalesce(opt.align, align) | 		align = langext.Coalesce(opt.align, align) | ||||||
| @@ -184,6 +224,7 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | |||||||
| 		fontNameOverride = langext.CoalesceOpt(opt.fontNameOverride, fontNameOverride) | 		fontNameOverride = langext.CoalesceOpt(opt.fontNameOverride, fontNameOverride) | ||||||
| 		fontStyleOverride = langext.CoalesceOpt(opt.fontStyleOverride, fontStyleOverride) | 		fontStyleOverride = langext.CoalesceOpt(opt.fontStyleOverride, fontStyleOverride) | ||||||
| 		fontSizeOverride = langext.CoalesceOpt(opt.fontSizeOverride, fontSizeOverride) | 		fontSizeOverride = langext.CoalesceOpt(opt.fontSizeOverride, fontSizeOverride) | ||||||
|  | 		alphaOverride = langext.CoalesceOpt(opt.alphaOverride, alphaOverride) | ||||||
| 		extraLn = langext.Coalesce(opt.extraLn, extraLn) | 		extraLn = langext.Coalesce(opt.extraLn, extraLn) | ||||||
| 		x = langext.CoalesceOpt(opt.x, x) | 		x = langext.CoalesceOpt(opt.x, x) | ||||||
| 		autoWidth = langext.Coalesce(opt.autoWidth, autoWidth) | 		autoWidth = langext.Coalesce(opt.autoWidth, autoWidth) | ||||||
| @@ -191,6 +232,7 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | |||||||
| 		borderColor = langext.CoalesceOpt(opt.borderColor, borderColor) | 		borderColor = langext.CoalesceOpt(opt.borderColor, borderColor) | ||||||
| 		fillColor = langext.CoalesceOpt(opt.fillColor, fillColor) | 		fillColor = langext.CoalesceOpt(opt.fillColor, fillColor) | ||||||
| 		autoWidthPaddingX = langext.Coalesce(opt.autoWidthPaddingX, autoWidthPaddingX) | 		autoWidthPaddingX = langext.Coalesce(opt.autoWidthPaddingX, autoWidthPaddingX) | ||||||
|  | 		debug = langext.Coalesce(opt.debug, debug) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if fontNameOverride != nil || fontStyleOverride != nil || fontSizeOverride != nil { | 	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) }() | 		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 { | 	if textColor != nil { | ||||||
| 		oldColorR, oldColorG, oldColorB := b.b.GetTextColor() | 		oldColorR, oldColorG, oldColorB := b.b.GetTextColor() | ||||||
| 		b.SetTextColor(textColor.R, textColor.G, textColor.B) | 		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) }() | 		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 { | 	if x != nil { | ||||||
| 		b.b.SetX(*x) | 		b.b.SetX(*x) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if autoWidth { | 	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()-xBefore-b.GetMarginRight(), *height, RectOutline, NewPDFRectOpt().X(xBefore).Y(yBefore).LineWidth(0.25).DrawColor(0, 128, 0)) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	if extraLn != 0 { | 	if extraLn != 0 { | ||||||
| 		b.b.Ln(extraLn) | 		b.b.Ln(extraLn) | ||||||
|   | |||||||
| @@ -74,6 +74,35 @@ const ( | |||||||
| 	RectFillOutline PDFRectStyle = "FD" | 	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 ( | const ( | ||||||
| 	BackgroundFill        = true | 	BackgroundFill        = true | ||||||
| 	BackgroundTransparent = false | 	BackgroundTransparent = false | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ package wpdf | |||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
| 	"github.com/jung-kurt/gofpdf" | 	"github.com/jung-kurt/gofpdf" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/imageext" | 	"gogs.mikescher.com/BlackForestBytes/goext/imageext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"image" | 	"image" | ||||||
| @@ -65,7 +66,12 @@ func (b *WPDFBuilder) RegisterImage(bin []byte, opts ...*PDFImageRegisterOpt) *P | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if imageType == "" { | 	if imageType == "" { | ||||||
| 		ct := http.DetectContentType(bin[:512]) | 		ct := "" | ||||||
|  | 		if len(bin) > 512 { | ||||||
|  | 			ct = http.DetectContentType(bin[:512]) | ||||||
|  | 		} else { | ||||||
|  | 			ct = http.DetectContentType(bin) | ||||||
|  | 		} | ||||||
| 		switch ct { | 		switch ct { | ||||||
| 		case "image/jpg": | 		case "image/jpg": | ||||||
| 			imageType = "JPG" | 			imageType = "JPG" | ||||||
| @@ -125,6 +131,8 @@ type PDFImageOpt struct { | |||||||
| 	compression           *imageext.ImageCompresson | 	compression           *imageext.ImageCompresson | ||||||
| 	reEncodePixelPerMM    *float64 | 	reEncodePixelPerMM    *float64 | ||||||
| 	crop                  *imageext.ImageCrop | 	crop                  *imageext.ImageCrop | ||||||
|  | 	alphaOverride         *dataext.Tuple[float64, PDFBlendMode] | ||||||
|  | 	debug                 *bool | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewPDFImageOpt() *PDFImageOpt { | func NewPDFImageOpt() *PDFImageOpt { | ||||||
| @@ -151,6 +159,11 @@ func (opt *PDFImageOpt) Height(v float64) *PDFImageOpt { | |||||||
| 	return opt | 	return opt | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (opt *PDFImageOpt) Debug(v bool) *PDFImageOpt { | ||||||
|  | 	opt.debug = &v | ||||||
|  | 	return opt | ||||||
|  | } | ||||||
|  |  | ||||||
| func (opt *PDFImageOpt) Flow(v bool) *PDFImageOpt { | func (opt *PDFImageOpt) Flow(v bool) *PDFImageOpt { | ||||||
| 	opt.flow = &v | 	opt.flow = &v | ||||||
| 	return opt | 	return opt | ||||||
| @@ -212,6 +225,11 @@ func (opt *PDFImageOpt) Crop(cropX float64, cropY float64, cropWidth float64, cr | |||||||
| 	return opt | 	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) { | func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | ||||||
| 	var err error | 	var err error | ||||||
|  |  | ||||||
| @@ -229,7 +247,9 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | |||||||
| 	var imageFit *imageext.ImageFit = nil | 	var imageFit *imageext.ImageFit = nil | ||||||
| 	var fillColor color.Color = color.Transparent | 	var fillColor color.Color = color.Transparent | ||||||
| 	compression := imageext.CompressionPNGSpeed | 	compression := imageext.CompressionPNGSpeed | ||||||
|  | 	debug := b.debug | ||||||
| 	var crop *imageext.ImageCrop = nil | 	var crop *imageext.ImageCrop = nil | ||||||
|  | 	var alphaOverride *dataext.Tuple[float64, PDFBlendMode] | ||||||
|  |  | ||||||
| 	for _, opt := range opts { | 	for _, opt := range opts { | ||||||
| 		x = langext.Coalesce(opt.x, x) | 		x = langext.Coalesce(opt.x, x) | ||||||
| @@ -247,10 +267,18 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | |||||||
| 		compression = langext.Coalesce(opt.compression, compression) | 		compression = langext.Coalesce(opt.compression, compression) | ||||||
| 		reEncodePixelPerMM = langext.Coalesce(opt.reEncodePixelPerMM, reEncodePixelPerMM) | 		reEncodePixelPerMM = langext.Coalesce(opt.reEncodePixelPerMM, reEncodePixelPerMM) | ||||||
| 		crop = langext.CoalesceOpt(opt.crop, crop) | 		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 | 	regName := img.Name | ||||||
|  |  | ||||||
|  | 	var subImageBounds *imageext.PercentageRectangle = nil | ||||||
|  |  | ||||||
| 	if imageFit != nil || fillColor != nil || crop != nil { | 	if imageFit != nil || fillColor != nil || crop != nil { | ||||||
|  |  | ||||||
| 		var dataimg image.Image | 		var dataimg image.Image | ||||||
| @@ -278,11 +306,14 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | |||||||
| 			pxw := w * pdfPixelPerMillimeter | 			pxw := w * pdfPixelPerMillimeter | ||||||
| 			pxh := h * 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 { | 			if err != nil { | ||||||
| 				b.b.SetError(err) | 				b.b.SetError(err) | ||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | 			subImageBounds = &dataImgRect | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if dataimg.ColorModel() != color.RGBAModel && dataimg.ColorModel() != color.NRGBAModel { | 		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{ | 	fpdfOpt := gofpdf.ImageOptions{ | ||||||
| 		ImageType:             imageType, | 		ImageType:             imageType, | ||||||
| 		ReadDpi:               readDpi, | 		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) | 	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 | 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 { | type PDFMultiCellOpt struct { | ||||||
| 	width             *float64 | 	width             *float64 | ||||||
| @@ -11,11 +14,13 @@ type PDFMultiCellOpt struct { | |||||||
| 	fontNameOverride  *PDFFontFamily | 	fontNameOverride  *PDFFontFamily | ||||||
| 	fontStyleOverride *PDFFontStyle | 	fontStyleOverride *PDFFontStyle | ||||||
| 	fontSizeOverride  *float64 | 	fontSizeOverride  *float64 | ||||||
|  | 	alphaOverride     *dataext.Tuple[float64, PDFBlendMode] | ||||||
| 	extraLn           *float64 | 	extraLn           *float64 | ||||||
| 	x                 *float64 | 	x                 *float64 | ||||||
| 	textColor         *PDFColor | 	textColor         *PDFColor | ||||||
| 	borderColor       *PDFColor | 	borderColor       *PDFColor | ||||||
| 	fillColor         *PDFColor | 	fillColor         *PDFColor | ||||||
|  | 	debug             *bool | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewPDFMultiCellOpt() *PDFMultiCellOpt { | func NewPDFMultiCellOpt() *PDFMultiCellOpt { | ||||||
| @@ -119,6 +124,21 @@ func (opt *PDFMultiCellOpt) FillColorHex(c uint32) *PDFMultiCellOpt { | |||||||
| 	return opt | 	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) { | func (b *WPDFBuilder) MultiCell(txt string, opts ...*PDFMultiCellOpt) { | ||||||
|  |  | ||||||
| 	txtTR := b.tr(txt) | 	txtTR := b.tr(txt) | ||||||
| @@ -131,11 +151,13 @@ func (b *WPDFBuilder) MultiCell(txt string, opts ...*PDFMultiCellOpt) { | |||||||
| 	var fontNameOverride *PDFFontFamily | 	var fontNameOverride *PDFFontFamily | ||||||
| 	var fontStyleOverride *PDFFontStyle | 	var fontStyleOverride *PDFFontStyle | ||||||
| 	var fontSizeOverride *float64 | 	var fontSizeOverride *float64 | ||||||
|  | 	var alphaOverride *dataext.Tuple[float64, PDFBlendMode] | ||||||
| 	extraLn := float64(0) | 	extraLn := float64(0) | ||||||
| 	var x *float64 | 	var x *float64 | ||||||
| 	var textColor *PDFColor | 	var textColor *PDFColor | ||||||
| 	var borderColor *PDFColor | 	var borderColor *PDFColor | ||||||
| 	var fillColor *PDFColor | 	var fillColor *PDFColor | ||||||
|  | 	debug := b.debug | ||||||
|  |  | ||||||
| 	for _, opt := range opts { | 	for _, opt := range opts { | ||||||
| 		width = langext.Coalesce(opt.width, width) | 		width = langext.Coalesce(opt.width, width) | ||||||
| @@ -146,11 +168,13 @@ func (b *WPDFBuilder) MultiCell(txt string, opts ...*PDFMultiCellOpt) { | |||||||
| 		fontNameOverride = langext.CoalesceOpt(opt.fontNameOverride, fontNameOverride) | 		fontNameOverride = langext.CoalesceOpt(opt.fontNameOverride, fontNameOverride) | ||||||
| 		fontStyleOverride = langext.CoalesceOpt(opt.fontStyleOverride, fontStyleOverride) | 		fontStyleOverride = langext.CoalesceOpt(opt.fontStyleOverride, fontStyleOverride) | ||||||
| 		fontSizeOverride = langext.CoalesceOpt(opt.fontSizeOverride, fontSizeOverride) | 		fontSizeOverride = langext.CoalesceOpt(opt.fontSizeOverride, fontSizeOverride) | ||||||
|  | 		alphaOverride = langext.CoalesceOpt(opt.alphaOverride, alphaOverride) | ||||||
| 		extraLn = langext.Coalesce(opt.extraLn, extraLn) | 		extraLn = langext.Coalesce(opt.extraLn, extraLn) | ||||||
| 		x = langext.CoalesceOpt(opt.x, x) | 		x = langext.CoalesceOpt(opt.x, x) | ||||||
| 		textColor = langext.CoalesceOpt(opt.textColor, textColor) | 		textColor = langext.CoalesceOpt(opt.textColor, textColor) | ||||||
| 		borderColor = langext.CoalesceOpt(opt.borderColor, borderColor) | 		borderColor = langext.CoalesceOpt(opt.borderColor, borderColor) | ||||||
| 		fillColor = langext.CoalesceOpt(opt.fillColor, fillColor) | 		fillColor = langext.CoalesceOpt(opt.fillColor, fillColor) | ||||||
|  | 		debug = langext.Coalesce(opt.debug, debug) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if fontNameOverride != nil || fontStyleOverride != nil || fontSizeOverride != nil { | 	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) }() | 		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 { | 	if x != nil { | ||||||
| 		b.b.SetX(*x) | 		b.b.SetX(*x) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	xBefore, yBefore := b.b.GetXY() | ||||||
|  |  | ||||||
| 	b.b.MultiCell(width, height, txtTR, string(border), string(align), fill) | 	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 { | 	if extraLn != 0 { | ||||||
| 		b.b.Ln(extraLn) | 		b.b.Ln(extraLn) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -1,6 +1,9 @@ | |||||||
| package wpdf | 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 { | type PDFRectOpt struct { | ||||||
| 	x         *float64 | 	x         *float64 | ||||||
| @@ -8,10 +11,12 @@ type PDFRectOpt struct { | |||||||
| 	lineWidth *float64 | 	lineWidth *float64 | ||||||
| 	drawColor *PDFColor | 	drawColor *PDFColor | ||||||
| 	fillColor *PDFColor | 	fillColor *PDFColor | ||||||
|  | 	alpha     *dataext.Tuple[float64, PDFBlendMode] | ||||||
| 	radiusTL  *float64 | 	radiusTL  *float64 | ||||||
| 	radiusTR  *float64 | 	radiusTR  *float64 | ||||||
| 	radiusBR  *float64 | 	radiusBR  *float64 | ||||||
| 	radiusBL  *float64 | 	radiusBL  *float64 | ||||||
|  | 	debug     *bool | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewPDFRectOpt() *PDFRectOpt { | func NewPDFRectOpt() *PDFRectOpt { | ||||||
| @@ -81,16 +86,28 @@ func (opt *PDFRectOpt) RadiusBR(radius float64) *PDFRectOpt { | |||||||
| 	return opt | 	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) { | func (b *WPDFBuilder) Rect(w float64, h float64, styleStr PDFRectStyle, opts ...*PDFRectOpt) { | ||||||
| 	x := b.GetX() | 	x := b.GetX() | ||||||
| 	y := b.GetY() | 	y := b.GetY() | ||||||
| 	var lineWidth *float64 | 	var lineWidth *float64 | ||||||
| 	var drawColor *PDFColor | 	var drawColor *PDFColor | ||||||
| 	var fillColor *PDFColor | 	var fillColor *PDFColor | ||||||
|  | 	var alphaOverride *dataext.Tuple[float64, PDFBlendMode] | ||||||
| 	radiusTL := float64(0) | 	radiusTL := float64(0) | ||||||
| 	radiusTR := float64(0) | 	radiusTR := float64(0) | ||||||
| 	radiusBR := float64(0) | 	radiusBR := float64(0) | ||||||
| 	radiusBL := float64(0) | 	radiusBL := float64(0) | ||||||
|  | 	debug := b.debug | ||||||
|  |  | ||||||
| 	for _, opt := range opts { | 	for _, opt := range opts { | ||||||
| 		x = langext.Coalesce(opt.x, x) | 		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) | 		lineWidth = langext.CoalesceOpt(opt.lineWidth, lineWidth) | ||||||
| 		drawColor = langext.CoalesceOpt(opt.drawColor, drawColor) | 		drawColor = langext.CoalesceOpt(opt.drawColor, drawColor) | ||||||
| 		fillColor = langext.CoalesceOpt(opt.fillColor, fillColor) | 		fillColor = langext.CoalesceOpt(opt.fillColor, fillColor) | ||||||
|  | 		alphaOverride = langext.CoalesceOpt(opt.alpha, alphaOverride) | ||||||
| 		radiusTL = langext.Coalesce(opt.radiusTL, radiusTL) | 		radiusTL = langext.Coalesce(opt.radiusTL, radiusTL) | ||||||
| 		radiusTR = langext.Coalesce(opt.radiusTR, radiusTR) | 		radiusTR = langext.Coalesce(opt.radiusTR, radiusTR) | ||||||
| 		radiusBR = langext.Coalesce(opt.radiusBR, radiusBR) | 		radiusBR = langext.Coalesce(opt.radiusBR, radiusBR) | ||||||
| 		radiusBL = langext.Coalesce(opt.radiusBL, radiusBL) | 		radiusBL = langext.Coalesce(opt.radiusBL, radiusBL) | ||||||
|  | 		debug = langext.Coalesce(opt.debug, debug) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if lineWidth != nil { | 	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) }() | 		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)) | 	b.b.RoundedRectExt(x, y, w, h, radiusTL, radiusTR, radiusBR, radiusBL, string(styleStr)) | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										351
									
								
								wpdf/wpdfTable.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										351
									
								
								wpdf/wpdfTable.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,351 @@ | |||||||
|  | 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 | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	{ | ||||||
|  | 		rmSub := 0.0 | ||||||
|  | 		for i := range columnDef { | ||||||
|  | 			if frColumnWeights[i] != 0 { | ||||||
|  | 				w := min(autoWidths[i], (remainingWidth/float64(frColumnWidthCount))*frColumnWeights[i]) | ||||||
|  | 				rmSub += w - columnWidths[i] | ||||||
|  | 				columnWidths[i] = w | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		remainingWidth -= rmSub | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if remainingWidth > 0.01 { | ||||||
|  | 		rmSub := 0.0 | ||||||
|  | 		for i, _ := range columnDef { | ||||||
|  | 			if frColumnWeights[i] != 0 { | ||||||
|  | 				addW := (remainingWidth / float64(frColumnWidthCount)) * frColumnWeights[i] | ||||||
|  | 				rmSub += addW | ||||||
|  | 				columnWidths[i] += addW | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		remainingWidth -= rmSub | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	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