gg/util.go

159 lines
3.0 KiB
Go
Raw Normal View History

2016-02-19 15:43:37 +01:00
package gg
2016-02-18 22:02:57 +01:00
import (
2016-02-19 17:07:25 +01:00
"fmt"
2016-02-18 22:02:57 +01:00
"image"
2016-02-19 17:07:25 +01:00
"image/draw"
2016-02-18 22:02:57 +01:00
"image/png"
2016-02-19 04:53:47 +01:00
"io/ioutil"
2016-02-19 19:16:40 +01:00
"math"
2016-02-18 22:02:57 +01:00
"os"
2016-02-19 17:07:25 +01:00
"strings"
2016-02-18 22:02:57 +01:00
2016-02-22 04:57:37 +01:00
"github.com/golang/freetype/raster"
2016-02-19 04:53:47 +01:00
"github.com/golang/freetype/truetype"
"golang.org/x/image/font"
2016-02-18 22:02:57 +01:00
"golang.org/x/image/math/fixed"
)
2016-02-19 19:16:40 +01:00
func Radians(degrees float64) float64 {
return degrees * math.Pi / 180
}
func Degrees(radians float64) float64 {
return radians * 180 / math.Pi
}
2016-02-19 19:49:03 +01:00
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 {
2016-02-18 22:02:57 +01:00
file, err := os.Create(path)
if err != nil {
return err
}
defer file.Close()
return png.Encode(file, im)
}
2016-02-19 17:07:25 +01:00
func imageToRGBA(src image.Image) *image.RGBA {
dst := image.NewRGBA(src.Bounds())
draw.Draw(dst, dst.Rect, src, image.ZP, draw.Src)
return dst
}
2016-02-20 21:42:36 +01:00
func parseHexColor(x string) (r, g, b, a int) {
2016-02-19 17:07:25 +01:00
x = strings.TrimPrefix(x, "#")
2016-02-20 21:42:36 +01:00
a = 255
2016-02-19 17:07:25 +01:00
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)
}
2016-02-20 21:42:36 +01:00
if len(x) == 8 {
format := "%02x%02x%02x%02x"
fmt.Sscanf(x, format, &r, &g, &b, &a)
}
2016-02-19 17:07:25 +01:00
return
}
2016-02-18 22:02:57 +01:00
func fp(x, y float64) fixed.Point26_6 {
return fixed.Point26_6{fixed.Int26_6(x * 64), fixed.Int26_6(y * 64)}
}
func fi(x float64) fixed.Int26_6 {
return fixed.Int26_6(x * 64)
}
2016-02-19 04:53:47 +01:00
2016-02-22 04:57:37 +01:00
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
}
2016-02-19 20:03:52 +01:00
func loadFontFace(path string, points float64) font.Face {
2016-02-19 04:53:47 +01:00
fontBytes, err := ioutil.ReadFile(path)
if err != nil {
panic(err)
}
f, err := truetype.Parse(fontBytes)
if err != nil {
panic(err)
}
return truetype.NewFace(f, &truetype.Options{
2016-02-20 05:03:39 +01:00
Size: points,
// Hinting: font.HintingFull,
2016-02-19 04:53:47 +01:00
})
}
2016-02-22 04:57:37 +01:00
func flattenPath(p raster.Path) [][]Point {
var result [][]Point
path := make([]Point, 0, 16)
var cx, cy float64
for i := 0; i < len(p); {
switch p[i] {
case 0:
if len(path) > 0 {
result = append(result, path)
path = make([]Point, 0, 16)
}
x := unfix(p[i+1])
y := unfix(p[i+2])
path = append(path, Point{x, y})
cx, cy = x, y
i += 4
case 1:
x := unfix(p[i+1])
y := unfix(p[i+2])
path = append(path, Point{x, y})
cx, cy = x, y
i += 4
case 2:
x1 := unfix(p[i+1])
y1 := unfix(p[i+2])
x2 := unfix(p[i+3])
y2 := unfix(p[i+4])
points := QuadraticBezier(cx, cy, x1, y1, x2, y2)
path = append(path, points...)
cx, cy = x2, y2
i += 6
case 3:
x1 := unfix(p[i+1])
y1 := unfix(p[i+2])
x2 := unfix(p[i+3])
y2 := unfix(p[i+4])
x3 := unfix(p[i+5])
y3 := unfix(p[i+6])
points := CubicBezier(cx, cy, x1, y1, x2, y2, x3, y3)
path = append(path, points...)
cx, cy = x3, y3
i += 8
default:
panic("bad path")
}
}
if len(path) > 0 {
result = append(result, path)
}
return result
}