image.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. package service
  2. import (
  3. "image"
  4. "image/color"
  5. "image/draw"
  6. "math"
  7. "github.com/golang/freetype"
  8. "github.com/golang/freetype/truetype"
  9. )
  10. // Image Image.
  11. type Image struct {
  12. *image.RGBA
  13. }
  14. // newImage create a new image
  15. func newImage(width, length int) *Image {
  16. img := &Image{image.NewRGBA(image.Rect(0, 0, width, length))}
  17. return img
  18. }
  19. func (img *Image) fillBkg(c image.Image) {
  20. draw.Draw(img, img.Bounds(), c, image.ZP, draw.Over)
  21. }
  22. // drawCircle draw circle.
  23. func (img *Image) drawCircle(cx, cy, radius int, isFillColor bool, color color.Color) {
  24. point := img.Bounds().Size()
  25. // 如果圆在图片可见区域外,直接退出
  26. if cx+radius < 0 || cx-radius >= point.X || cy+radius < 0 || cy-radius >= point.Y {
  27. return
  28. }
  29. x, y, d := 0, radius, 3-2*radius
  30. for x <= y {
  31. if isFillColor {
  32. for yi := x; yi <= y; yi++ {
  33. img.drawCircle8(cx, cy, x, yi, color)
  34. }
  35. } else {
  36. img.drawCircle8(cx, cy, x, y, color)
  37. }
  38. if d < 0 {
  39. d = d + 4*x + 6
  40. } else {
  41. d = d + 4*(x-y) + 10
  42. y--
  43. }
  44. x++
  45. }
  46. }
  47. // drawLine .
  48. // Bresenham算法(https://zh.wikipedia.org/zh-cn/布雷森漢姆直線演算法).
  49. // startX,startY 起点 endX,endY终点.
  50. func (img *Image) drawLine(startX, startY, endX, endY int, color color.Color) {
  51. dx, dy, flag := int(math.Abs(float64(startY-startX))), int(math.Abs(float64(endY-startY))), false
  52. if dy > dx {
  53. flag = true
  54. startX, startY = startY, startX
  55. endX, endY = endY, endX
  56. dx, dy = dy, dx
  57. }
  58. ix, iy := sign(endX-startX), sign(endY-startY)
  59. n2dy := dy * 2
  60. n2dydx := (dy - dx) * 2
  61. d := n2dy - dx
  62. for startX != endX {
  63. if d < 0 {
  64. d += n2dy
  65. } else {
  66. startY += iy
  67. d += n2dydx
  68. }
  69. if flag {
  70. img.Set(startY, startX, color)
  71. } else {
  72. img.Set(startX, startY, color)
  73. }
  74. startX += ix
  75. }
  76. }
  77. func (img *Image) drawCircle8(xc, yc, x, y int, color color.Color) {
  78. img.Set(xc+x, yc+y, color)
  79. img.Set(xc-x, yc+y, color)
  80. img.Set(xc+x, yc-y, color)
  81. img.Set(xc-x, yc-y, color)
  82. img.Set(xc+y, yc+x, color)
  83. img.Set(xc-y, yc+x, color)
  84. img.Set(xc+y, yc-x, color)
  85. img.Set(xc-y, yc-x, color)
  86. }
  87. // drawString image draw string.
  88. func (img *Image) drawString(font *truetype.Font, color color.Color, str string, fontsize float64) {
  89. ctx := freetype.NewContext()
  90. // default 72dpi
  91. ctx.SetDst(img)
  92. ctx.SetClip(img.Bounds())
  93. ctx.SetSrc(image.NewUniform(color))
  94. ctx.SetFontSize(fontsize)
  95. ctx.SetFont(font)
  96. // 写入文字的位置
  97. pt := freetype.Pt(0, int(-fontsize/6)+ctx.PointToFixed(fontsize).Ceil())
  98. ctx.DrawString(str, pt)
  99. }
  100. // 水波纹, amplude=振幅, period=周期
  101. // copy from https://github.com/dchest/captcha/blob/master/image.go
  102. func (img *Image) distortTo(amplude float64, period float64) {
  103. w := img.Bounds().Max.X
  104. h := img.Bounds().Max.Y
  105. oldm := img.RGBA
  106. dx := 1.4 * math.Pi / period
  107. for x := 0; x < w; x++ {
  108. for y := 0; y < h; y++ {
  109. xo := amplude * math.Sin(float64(y)*dx)
  110. yo := amplude * math.Cos(float64(x)*dx)
  111. rgba := oldm.RGBAAt(x+int(xo), y+int(yo))
  112. if rgba.A > 0 {
  113. oldm.SetRGBA(x, y, rgba)
  114. }
  115. }
  116. }
  117. }
  118. // Rotate 旋转
  119. func (img *Image) rotate(angle float64) image.Image {
  120. return new(rotate).rotate(angle, img.RGBA).transformRGBA()
  121. }