123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132 |
- package service
- import (
- "image"
- "image/color"
- "image/draw"
- "math"
- "github.com/golang/freetype"
- "github.com/golang/freetype/truetype"
- )
- // Image Image.
- type Image struct {
- *image.RGBA
- }
- // newImage create a new image
- func newImage(width, length int) *Image {
- img := &Image{image.NewRGBA(image.Rect(0, 0, width, length))}
- return img
- }
- func (img *Image) fillBkg(c image.Image) {
- draw.Draw(img, img.Bounds(), c, image.ZP, draw.Over)
- }
- // drawCircle draw circle.
- func (img *Image) drawCircle(cx, cy, radius int, isFillColor bool, color color.Color) {
- point := img.Bounds().Size()
- // 如果圆在图片可见区域外,直接退出
- if cx+radius < 0 || cx-radius >= point.X || cy+radius < 0 || cy-radius >= point.Y {
- return
- }
- x, y, d := 0, radius, 3-2*radius
- for x <= y {
- if isFillColor {
- for yi := x; yi <= y; yi++ {
- img.drawCircle8(cx, cy, x, yi, color)
- }
- } else {
- img.drawCircle8(cx, cy, x, y, color)
- }
- if d < 0 {
- d = d + 4*x + 6
- } else {
- d = d + 4*(x-y) + 10
- y--
- }
- x++
- }
- }
- // drawLine .
- // Bresenham算法(https://zh.wikipedia.org/zh-cn/布雷森漢姆直線演算法).
- // startX,startY 起点 endX,endY终点.
- func (img *Image) drawLine(startX, startY, endX, endY int, color color.Color) {
- dx, dy, flag := int(math.Abs(float64(startY-startX))), int(math.Abs(float64(endY-startY))), false
- if dy > dx {
- flag = true
- startX, startY = startY, startX
- endX, endY = endY, endX
- dx, dy = dy, dx
- }
- ix, iy := sign(endX-startX), sign(endY-startY)
- n2dy := dy * 2
- n2dydx := (dy - dx) * 2
- d := n2dy - dx
- for startX != endX {
- if d < 0 {
- d += n2dy
- } else {
- startY += iy
- d += n2dydx
- }
- if flag {
- img.Set(startY, startX, color)
- } else {
- img.Set(startX, startY, color)
- }
- startX += ix
- }
- }
- func (img *Image) drawCircle8(xc, yc, x, y int, color color.Color) {
- img.Set(xc+x, yc+y, color)
- img.Set(xc-x, yc+y, color)
- img.Set(xc+x, yc-y, color)
- img.Set(xc-x, yc-y, color)
- img.Set(xc+y, yc+x, color)
- img.Set(xc-y, yc+x, color)
- img.Set(xc+y, yc-x, color)
- img.Set(xc-y, yc-x, color)
- }
- // drawString image draw string.
- func (img *Image) drawString(font *truetype.Font, color color.Color, str string, fontsize float64) {
- ctx := freetype.NewContext()
- // default 72dpi
- ctx.SetDst(img)
- ctx.SetClip(img.Bounds())
- ctx.SetSrc(image.NewUniform(color))
- ctx.SetFontSize(fontsize)
- ctx.SetFont(font)
- // 写入文字的位置
- pt := freetype.Pt(0, int(-fontsize/6)+ctx.PointToFixed(fontsize).Ceil())
- ctx.DrawString(str, pt)
- }
- // 水波纹, amplude=振幅, period=周期
- // copy from https://github.com/dchest/captcha/blob/master/image.go
- func (img *Image) distortTo(amplude float64, period float64) {
- w := img.Bounds().Max.X
- h := img.Bounds().Max.Y
- oldm := img.RGBA
- dx := 1.4 * math.Pi / period
- for x := 0; x < w; x++ {
- for y := 0; y < h; y++ {
- xo := amplude * math.Sin(float64(y)*dx)
- yo := amplude * math.Cos(float64(x)*dx)
- rgba := oldm.RGBAAt(x+int(xo), y+int(yo))
- if rgba.A > 0 {
- oldm.SetRGBA(x, y, rgba)
- }
- }
- }
- }
- // Rotate 旋转
- func (img *Image) rotate(angle float64) image.Image {
- return new(rotate).rotate(angle, img.RGBA).transformRGBA()
- }
|