// Copyright ©2022 The gg Authors. All rights reserved. // Use of this source code is governed by a MIT-style // license that can be found in the LICENSE file. package gg import ( "fmt" "image" "image/draw" "image/jpeg" "image/png" "io/fs" "math" "os" "path/filepath" "strings" "golang.org/x/image/font" "golang.org/x/image/font/opentype" "golang.org/x/image/math/fixed" ) func Radians(degrees float64) float64 { return degrees * math.Pi / 180 } func Degrees(radians float64) float64 { return radians * 180 / math.Pi } func LoadImage(path string) (image.Image, error) { file, err := os.Open(path) if err != nil { return nil, err } defer file.Close() im, _, err := image.Decode(file) return im, err } func LoadPNG(path string) (image.Image, error) { file, err := os.Open(path) if err != nil { return nil, err } defer file.Close() return png.Decode(file) } func SavePNG(path string, im image.Image) error { file, err := os.Create(path) if err != nil { return err } defer file.Close() err = png.Encode(file, im) if err != nil { return fmt.Errorf("could not encode PNG to %q: %w", path, err) } return file.Close() } func LoadJPG(path string) (image.Image, error) { file, err := os.Open(path) if err != nil { return nil, err } defer file.Close() return jpeg.Decode(file) } func SaveJPG(path string, im image.Image, quality int) error { file, err := os.Create(path) if err != nil { return err } defer file.Close() var opt jpeg.Options opt.Quality = quality err = jpeg.Encode(file, im, &opt) if err != nil { return fmt.Errorf("could not encode JPG to %q: %w", path, err) } return file.Close() } func imageToRGBA(src image.Image) *image.RGBA { bounds := src.Bounds() dst := image.NewRGBA(bounds) draw.Draw(dst, bounds, src, bounds.Min, draw.Src) return dst } func parseHexColor(x string) (r, g, b, a int) { x = strings.TrimPrefix(x, "#") a = 255 if len(x) == 3 { format := "%1x%1x%1x" fmt.Sscanf(x, format, &r, &g, &b) r |= r << 4 g |= g << 4 b |= b << 4 } if len(x) == 6 { format := "%02x%02x%02x" fmt.Sscanf(x, format, &r, &g, &b) } if len(x) == 8 { format := "%02x%02x%02x%02x" fmt.Sscanf(x, format, &r, &g, &b, &a) } return } func fixp(x, y float64) fixed.Point26_6 { return fixed.Point26_6{fix(x), fix(y)} } func fix(x float64) fixed.Int26_6 { return fixed.Int26_6(math.Round(x * 64)) } func unfix(x fixed.Int26_6) float64 { const shift, mask = 6, 1<<6 - 1 if x >= 0 { return float64(x>>shift) + float64(x&mask)/64 } x = -x if x >= 0 { return -(float64(x>>shift) + float64(x&mask)/64) } return 0 } // LoadFontFace is a helper function to load the specified font file with // the specified point size. Note that the returned `font.Face` objects // are not thread safe and cannot be used in parallel across goroutines. // You can usually just use the Context.LoadFontFace function instead of // this package-level function. func LoadFontFace(path string, points float64) (font.Face, error) { fontBytes, err := os.ReadFile(path) if err != nil { return nil, err } return LoadFontFaceFromBytes(fontBytes, points) } // LoadFontFaceFromFS is a helper function to load the specified font file from // the provided filesystem and path, with the specified point size. // // Note that the returned `font.Face` objects are not thread safe and // cannot be used in parallel across goroutines. // You can usually just use the Context.LoadFontFace function instead of // this package-level function. func LoadFontFaceFromFS(fsys fs.FS, path string, points float64) (font.Face, error) { if fsys == nil { switch { case filepath.IsAbs(path): var ( err error orig = path root = filepath.FromSlash("/") ) path, err = filepath.Rel(root, path) if err != nil { return nil, fmt.Errorf("could not find relative path for %q from %q: %w", orig, root, err) } fsys = os.DirFS(root) default: fsys = os.DirFS(".") } } fontBytes, err := fs.ReadFile(fsys, path) if err != nil { return nil, err } return LoadFontFaceFromBytes(fontBytes, points) } // LoadFontFace is a helper function to load the specified font with // the specified point size. Note that the returned `font.Face` objects // are not thread safe and cannot be used in parallel across goroutines. // You can usually just use the Context.LoadFontFace function instead of // this package-level function. func LoadFontFaceFromBytes(raw []byte, points float64) (font.Face, error) { f, err := opentype.Parse(raw) if err != nil { return nil, err } face, err := opentype.NewFace(f, &opentype.FaceOptions{ Size: points, DPI: 72, // Hinting: font.HintingFull, }) return face, err }