UNPKG

ng-captcha

Version:

A pure front-end angular captcha module.

185 lines 17.8 kB
/** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ import { Component, ViewChild, Output, EventEmitter, Input } from '@angular/core'; var CaptchaComponent = /** @class */ (function () { function CaptchaComponent() { this.generateCode = new EventEmitter(); this.canvas = document.createElement('canvas'); this.letters = ["0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z", "A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z"]; this.code = ''; } /** * @return {?} */ CaptchaComponent.prototype.ngOnInit = /** * @return {?} */ function () { this.init(); this.refresh(); }; /** * @return {?} */ CaptchaComponent.prototype.onClick = /** * @return {?} */ function () { this.refresh(); }; /** * @return {?} */ CaptchaComponent.prototype.init = /** * @return {?} */ function () { this.canvas.width = this.width || 100; this.canvas.height = this.height || 50; this.canvas.style.cursor = 'pointer'; this.canvas.innerHTML = '您的浏览器版本不支持canvas'; this.container.nativeElement.appendChild(this.canvas); }; // generate code // generate code /** * @return {?} */ CaptchaComponent.prototype.refresh = // generate code /** * @return {?} */ function () { this.code = ''; /** @type {?} */ var ctx = this.canvas.getContext('2d'); ctx.textBaseline = 'middle'; ctx.fillStyle = this.randomColor(180, 240); ctx.fillRect(0, 0, this.canvas.width, this.canvas.height); for (var i = 0; i < 4; i++) { /** @type {?} */ var letter = this.letters[this.randomNum(0, this.letters.length)]; this.code += letter; ctx.font = this.randomNum(this.canvas.height / 2, this.canvas.height) + 'px SimHei'; ctx.fillStyle = this.randomColor(50, 160); ctx.shadowOffsetX = this.randomNum(-3, 3); ctx.shadowOffsetY = this.randomNum(-3, 3); ctx.shadowBlur = this.randomNum(-3, 3); ctx.shadowColor = 'rgb(0, 0, 0, 0.3)'; /** @type {?} */ var x = this.canvas.width / 5 * i; /** @type {?} */ var y = this.canvas.height / 2; /** @type {?} */ var deg = this.randomNum(-30, 30) // setting rotate and origin point ; // setting rotate and origin point ctx.translate(x, y); ctx.rotate(deg * Math.PI / 180); ctx.fillText(letter, 0, 0); // reset rotate and origin point ctx.rotate(-deg * Math.PI / 180); ctx.translate(-x, -y); } // draw interfering line for (var i = 0; i < 4; i++) { ctx.strokeStyle = this.randomColor(40, 180); ctx.beginPath(); ctx.moveTo(this.randomNum(0, this.canvas.width), this.randomNum(0, this.canvas.height)); ctx.lineTo(this.randomNum(0, this.canvas.width), this.randomNum(0, this.canvas.height)); ctx.stroke(); } // draw interfering point for (var i = 0; i < this.canvas.width / 4; i++) { ctx.fillStyle = this.randomColor(); ctx.beginPath(); ctx.arc(this.randomNum(0, this.canvas.width), this.randomNum(0, this.canvas.height), 1, 0, 2 * Math.PI); ctx.fill(); } this.generateCode.emit(this.code.toLowerCase()); }; /** * return a random integer * @param min 最小值 * @param max 最大值 */ /** * return a random integer * @param {?} min 最小值 * @param {?} max 最大值 * @return {?} */ CaptchaComponent.prototype.randomNum = /** * return a random integer * @param {?} min 最小值 * @param {?} max 最大值 * @return {?} */ function (min, max) { return Math.floor(Math.random() * (max - min) + min); }; /** * return a random color */ /** * return a random color * @param {?=} min * @param {?=} max * @return {?} */ CaptchaComponent.prototype.randomColor = /** * return a random color * @param {?=} min * @param {?=} max * @return {?} */ function (min, max) { if (min === void 0) { min = 0; } if (max === void 0) { max = 255; } /** @type {?} */ var r = this.randomNum(min, max); /** @type {?} */ var g = this.randomNum(min, max); /** @type {?} */ var b = this.randomNum(min, max); return "rgb(" + r + ", " + g + ", " + b + ")"; }; CaptchaComponent.decorators = [ { type: Component, args: [{ selector: 'captcha', template: "<div class=\"captcha-container\" #captchaContainer (click)=\"onClick()\"></div>", styles: [".captcha-container{display:inline-block;line-height:0}"] }] } ]; /** @nocollapse */ CaptchaComponent.ctorParameters = function () { return []; }; CaptchaComponent.propDecorators = { width: [{ type: Input }], height: [{ type: Input }], generateCode: [{ type: Output }], container: [{ type: ViewChild, args: ['captchaContainer',] }] }; return CaptchaComponent; }()); export { CaptchaComponent }; if (false) { /** @type {?} */ CaptchaComponent.prototype.width; /** @type {?} */ CaptchaComponent.prototype.height; /** @type {?} */ CaptchaComponent.prototype.generateCode; /** @type {?} */ CaptchaComponent.prototype.container; /** @type {?} */ CaptchaComponent.prototype.canvas; /** @type {?} */ CaptchaComponent.prototype.letters; /** @type {?} */ CaptchaComponent.prototype.code; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"captcha.component.js","sourceRoot":"ng://ng-captcha/","sources":["lib/captcha.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, ViewChild, Output, EventEmitter, Input } from '@angular/core';\n\n@Component({\n  selector: 'captcha',\n  templateUrl: './captcha.component.html',\n  styleUrls: ['./captcha.component.css']\n})\nexport class CaptchaComponent implements OnInit {\n\n  @Input() width\n  @Input() height\n  @Output() generateCode = new EventEmitter()\n  @ViewChild('captchaContainer') container\n  canvas = document.createElement('canvas')\n  letters = [\"0\", \"1\", \"2\", \"3\", \"4\", \"5\", \"6\", \"7\", \"8\", \"9\", \"a\", \"b\", \"c\", \"d\", \"e\", \"f\", \"g\", \"h\", \"i\", \"j\", \"k\", \"l\", \"m\", \"n\", \"o\", \"p\", \"q\", \"r\", \"s\", \"t\", \"u\", \"v\", \"w\", \"x\", \"y\", \"z\", \"A\", \"B\", \"C\", \"D\", \"E\", \"F\", \"G\", \"H\", \"I\", \"J\", \"K\", \"L\", \"M\", \"N\", \"O\", \"P\", \"Q\", \"R\", \"S\", \"T\", \"U\", \"V\", \"W\", \"X\", \"Y\", \"Z\"]\n  code = ''\n\n  constructor() { }\n\n  ngOnInit() {\n    this.init()\n    this.refresh()\n  }\n\n  onClick() {\n    this.refresh()\n  }\n\n  private init() {\n    this.canvas.width = this.width || 100\n    this.canvas.height = this.height || 50\n    this.canvas.style.cursor = 'pointer'\n    this.canvas.innerHTML = '您的浏览器版本不支持canvas'\n    this.container.nativeElement.appendChild(this.canvas)\n  }\n\n  // generate code\n  private refresh() {\n    this.code = ''\n    const ctx = this.canvas.getContext('2d')\n    ctx.textBaseline = 'middle'\n    ctx.fillStyle = this.randomColor(180, 240)\n    ctx.fillRect(0, 0, this.canvas.width, this.canvas.height)\n\n    for (let i = 0; i < 4; i++) {\n      const letter = this.letters[this.randomNum(0, this.letters.length)]\n      this.code += letter\n      ctx.font = this.randomNum(this.canvas.height / 2, this.canvas.height) + 'px SimHei'\n      ctx.fillStyle = this.randomColor(50, 160)\n      ctx.shadowOffsetX = this.randomNum(-3, 3)\n      ctx.shadowOffsetY = this.randomNum(-3, 3)\n      ctx.shadowBlur = this.randomNum(-3, 3)\n      ctx.shadowColor = 'rgb(0, 0, 0, 0.3)'\n      const x = this.canvas.width / 5 * i\n      const y = this.canvas.height / 2\n      const deg = this.randomNum(-30, 30)\n      // setting rotate and origin point\n      ctx.translate(x, y)\n      ctx.rotate(deg * Math.PI / 180)\n      ctx.fillText(letter, 0, 0)\n      // reset rotate and origin point\n      ctx.rotate(-deg * Math.PI / 180)\n      ctx.translate(-x, -y)\n    }\n\n    // draw interfering line\n    for (let i = 0; i < 4; i++) {\n      ctx.strokeStyle = this.randomColor(40, 180)\n      ctx.beginPath()\n      ctx.moveTo(this.randomNum(0, this.canvas.width), this.randomNum(0, this.canvas.height))\n      ctx.lineTo(this.randomNum(0, this.canvas.width), this.randomNum(0, this.canvas.height))\n      ctx.stroke()\n    }\n\n    // draw interfering point\n    for (let i = 0; i < this.canvas.width / 4; i++) {\n      ctx.fillStyle = this.randomColor()\n      ctx.beginPath()\n      ctx.arc(this.randomNum(0, this.canvas.width), this.randomNum(0, this.canvas.height), 1, 0, 2 * Math.PI)\n      ctx.fill()\n    }\n\n    this.generateCode.emit(this.code.toLowerCase())\n  }\n\n  /**\n   * return a random integer\n   * @param min 最小值\n   * @param max 最大值\n   */\n  private randomNum(min, max) {\n    return Math.floor(Math.random() * (max - min) + min)\n  }\n\n  /**\n   * return a random color\n   */\n  private randomColor(min = 0, max = 255) {\n    const r = this.randomNum(min, max)\n    const g = this.randomNum(min, max)\n    const b = this.randomNum(min, max)\n    return `rgb(${r}, ${g}, ${b})`\n  }\n\n}\n"]}