draw.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. package service
  2. import (
  3. "image"
  4. "image/color"
  5. "image/draw"
  6. "io/ioutil"
  7. "math"
  8. "math/rand"
  9. "time"
  10. "go-common/app/interface/main/captcha/conf"
  11. "github.com/golang/freetype"
  12. "github.com/golang/freetype/truetype"
  13. )
  14. // CONST VALUE.
  15. const (
  16. NORMAL = int(4)
  17. MEDIUM = int(8)
  18. HIGH = int(16)
  19. MinLenStart = int(4)
  20. MinWidth = int(48)
  21. MinLength = int(20)
  22. Length48 = int(48)
  23. TypeNone = int(0)
  24. TypeLOWER = int(1)
  25. TypeUPPER = int(2)
  26. TypeALL = int(3)
  27. )
  28. var fontKinds = [][]int{[]int{10, 48}, []int{26, 97}, []int{26, 65}}
  29. func sign(x int) int {
  30. if x > 0 {
  31. return 1
  32. }
  33. return -1
  34. }
  35. // NewCaptcha new a captcha.
  36. func newCaptcha(c *conf.Captcha) *Captcha {
  37. captcha := &Captcha{
  38. disturbLevel: NORMAL,
  39. }
  40. captcha.frontColors = []color.Color{color.Black}
  41. captcha.bkgColors = []color.Color{color.White}
  42. captcha.setFont(c.Fonts...)
  43. colors := []color.Color{}
  44. for _, v := range c.BkgColors {
  45. colors = append(colors, v)
  46. }
  47. captcha.setBkgColor(colors...)
  48. colors = []color.Color{}
  49. for _, v := range c.FrontColors {
  50. colors = append(colors, v)
  51. }
  52. captcha.setFontColor(colors...)
  53. captcha.setDisturbance(c.DisturbLevel)
  54. return captcha
  55. }
  56. // addFont add font.
  57. func (c *Captcha) addFont(path string) error {
  58. fontdata, erro := ioutil.ReadFile(path)
  59. if erro != nil {
  60. return erro
  61. }
  62. font, erro := freetype.ParseFont(fontdata)
  63. if erro != nil {
  64. return erro
  65. }
  66. if c.fonts == nil {
  67. c.fonts = []*truetype.Font{}
  68. }
  69. c.fonts = append(c.fonts, font)
  70. return nil
  71. }
  72. // setFont set font.
  73. func (c *Captcha) setFont(paths ...string) (err error) {
  74. for _, v := range paths {
  75. if err = c.addFont(v); err != nil {
  76. return err
  77. }
  78. }
  79. return nil
  80. }
  81. // setBkgColor set backgroud color.
  82. func (c *Captcha) setBkgColor(colors ...color.Color) {
  83. if len(colors) > 0 {
  84. c.bkgColors = c.bkgColors[:0]
  85. c.bkgColors = append(c.bkgColors, colors...)
  86. }
  87. }
  88. func (c *Captcha) randFont() *truetype.Font {
  89. return c.fonts[rand.Intn(len(c.fonts))]
  90. }
  91. // setBkgsetFontColorColor set font color.
  92. func (c *Captcha) setFontColor(colors ...color.Color) {
  93. if len(colors) > 0 {
  94. c.frontColors = c.frontColors[:0]
  95. c.frontColors = append(c.frontColors, colors...)
  96. }
  97. }
  98. // setDisturbance set disturbance.
  99. func (c *Captcha) setDisturbance(d int) {
  100. if d > 0 {
  101. c.disturbLevel = d
  102. }
  103. }
  104. func (c *Captcha) createImage(lenStart, lenEnd, width, length, t int) (image *Image, str string) {
  105. num := MinLenStart
  106. if lenStart < MinLenStart {
  107. lenStart = MinLenStart
  108. }
  109. if lenEnd > lenStart {
  110. // rand.Seed(time.Now().UnixNano())
  111. num = rand.Intn(lenEnd-lenStart+1) + lenStart
  112. }
  113. str = c.randStr(num, t)
  114. return c.createCustom(str, width, length), str
  115. }
  116. func (c *Captcha) createCustom(str string, width, length int) *Image {
  117. // boundary check
  118. if len(str) == 0 {
  119. str = "bilibili"
  120. }
  121. if width < MinWidth {
  122. width = MinWidth
  123. }
  124. if length < MinLength {
  125. length = MinLength
  126. }
  127. dst := newImage(width, length)
  128. c.drawBkg(dst)
  129. c.drawNoises(dst)
  130. c.drawString(dst, str, width, length)
  131. return dst
  132. }
  133. // randStr ascII random
  134. // 48~57 -> 0~9 number
  135. // 65~90 -> A~Z uppercase
  136. // 98~122 -> a~z lowcase
  137. func (c *Captcha) randStr(size, kind int) string {
  138. ikind, result := kind, make([]byte, size)
  139. isAll := kind > TypeUPPER || kind < TypeNone
  140. // rand.Seed(time.Now().UnixNano())
  141. for i := 0; i < size; i++ {
  142. if isAll {
  143. ikind = rand.Intn(TypeALL)
  144. }
  145. scope, base := fontKinds[ikind][0], fontKinds[ikind][1]
  146. result[i] = uint8(base + rand.Intn(scope))
  147. }
  148. return string(result)
  149. }
  150. func (c *Captcha) drawBkg(img *Image) {
  151. ra := rand.New(rand.NewSource(time.Now().UnixNano()))
  152. //填充主背景色
  153. bgcolorindex := ra.Intn(len(c.bkgColors))
  154. bkg := image.NewUniform(c.bkgColors[bgcolorindex])
  155. img.fillBkg(bkg)
  156. }
  157. func (c *Captcha) drawNoises(img *Image) {
  158. ra := rand.New(rand.NewSource(time.Now().UnixNano()))
  159. //// 待绘制图片的尺寸
  160. point := img.Bounds().Size()
  161. disturbLevel := c.disturbLevel
  162. // 绘制干扰斑点
  163. for i := 0; i < disturbLevel; i++ {
  164. x := ra.Intn(point.X)
  165. y := ra.Intn(point.Y)
  166. radius := ra.Intn(point.Y/20) + 1
  167. colorindex := ra.Intn(len(c.frontColors))
  168. img.drawCircle(x, y, radius, i%4 != 0, c.frontColors[colorindex])
  169. }
  170. // 绘制干扰线
  171. for i := 0; i < disturbLevel; i++ {
  172. x := ra.Intn(point.X)
  173. y := ra.Intn(point.Y)
  174. o := int(math.Pow(-1, float64(i)))
  175. w := ra.Intn(point.Y) * o
  176. h := ra.Intn(point.Y/10) * o
  177. colorindex := ra.Intn(len(c.frontColors))
  178. img.drawLine(x, y, x+w, y+h, c.frontColors[colorindex])
  179. colorindex++
  180. }
  181. }
  182. // 绘制文字
  183. func (c *Captcha) drawString(img *Image, str string, width, length int) {
  184. if c.fonts == nil {
  185. panic("没有设置任何字体")
  186. }
  187. tmp := newImage(width, length)
  188. // 文字大小为图片高度的 0.6
  189. fsize := int(float64(length) * 0.6)
  190. // 用于生成随机角度
  191. r := rand.New(rand.NewSource(time.Now().UnixNano()))
  192. // 文字之间的距离
  193. // 左右各留文字的1/4大小为内部边距
  194. padding := fsize / 4
  195. gap := (width - padding*2) / (len(str))
  196. // 逐个绘制文字到图片上
  197. for i, char := range str {
  198. // 创建单个文字图片
  199. // 以文字为尺寸创建正方形的图形
  200. str := newImage(fsize, fsize)
  201. // str.FillBkg(image.NewUniform(color.Black))
  202. // 随机取一个前景色
  203. colorindex := r.Intn(len(c.frontColors))
  204. //随机取一个字体
  205. font := c.randFont()
  206. str.drawString(font, c.frontColors[colorindex], string(char), float64(fsize))
  207. // 转换角度后的文字图形
  208. rs := str.rotate(float64(r.Intn(40) - 20))
  209. // 计算文字位置
  210. s := rs.Bounds().Size()
  211. left := i*gap + padding
  212. top := (length - s.Y) / 2
  213. // 绘制到图片上
  214. draw.Draw(tmp, image.Rect(left, top, left+s.X, top+s.Y), rs, image.ZP, draw.Over)
  215. }
  216. if length >= Length48 {
  217. // 高度大于48添加波纹 小于48波纹影响用户识别
  218. tmp.distortTo(float64(fsize)/10, 200.0)
  219. }
  220. draw.Draw(img, tmp.Bounds(), tmp, image.ZP, draw.Over)
  221. }