UNPKG

17.6 kBJavaScriptView Raw
1/**
2 * SVG captcha.
3 * DO NOT use svg base captcha in production, since it may be parsable by bots.
4 * You need to convert to bitmap like png beforehand.
5 * @class ApCaptchaSvg
6 */
7
8'use strict';
9
10Object.defineProperty(exports, "__esModule", {
11 value: true
12});
13
14var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
15
16var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
17
18var _react = require('react');
19
20var _react2 = _interopRequireDefault(_react);
21
22var _randomval = require('randomval');
23
24var _randomval2 = _interopRequireDefault(_randomval);
25
26function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
27
28function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
29
30function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
31
32function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
33
34/** @lends ApCaptchaSvg */
35var ApCaptchaSvg = function (_Component) {
36 _inherits(ApCaptchaSvg, _Component);
37
38 function ApCaptchaSvg() {
39 _classCallCheck(this, ApCaptchaSvg);
40
41 return _possibleConstructorReturn(this, Object.getPrototypeOf(ApCaptchaSvg).apply(this, arguments));
42 }
43
44 _createClass(ApCaptchaSvg, [{
45 key: 'render',
46 value: function render() {
47 var s = this;
48 var props = s.props;
49
50
51 var color = '#555';
52
53 var width = props.width;
54 var height = props.height;
55
56
57 var backgrounds = [0, 1, 2, 3].map(function (val, i, _ref) {
58 var length = _ref.length;
59
60 var margin = width * 0.2;
61 var rate = (i + 0.5) / length;
62 return s._stripeBlock(rate, color, {
63 key: 'bg-' + i,
64 width: parseInt(width / length + margin * 2),
65 height: parseInt(height + margin * 2),
66 x: parseInt(width * rate - margin),
67 y: parseInt(0 - margin)
68 });
69 });
70
71 var texts = props.text.split('').map(function (letter, i, letters) {
72 var indices = [0, 1, 2, 3, 4, 5];
73 var real = _randomval2.default.randomInt(0, indices.length - 1);
74 var texts = indices.map(function (j) {
75 var rate = i / letters.length;
76 var key = 'letter-' + i + '-' + j;
77 if (j === real) {
78 return s.renderLetter(letter, rate, {
79 key: key,
80 fill: color
81 });
82 } else {
83 return s.renderLetter(s._dummyLetter(), rate, {
84 key: key,
85 fill: 'rgba(255,255,255,' + 0.01 * _randomval2.default.randomInt(0, 30) + ')'
86 });
87 }
88 });
89 return _react2.default.createElement(
90 'g',
91 { key: 'letter-group-' + i },
92 texts
93 );
94 });
95 return _react2.default.createElement(
96 'svg',
97 { version: props.version,
98 width: width,
99 height: height,
100 xmlns: props.xmlns,
101 viewBox: '0 0 ' + width + ' ' + height
102 },
103 _react2.default.createElement(
104 'g',
105 null,
106 backgrounds
107 ),
108 _react2.default.createElement(
109 'g',
110 null,
111 texts
112 )
113 );
114 }
115
116 // --------------------
117 // Specs
118 // --------------------
119
120 }, {
121 key: 'renderLetter',
122 value: function renderLetter(letter, rate, textProps) {
123 var s = this;
124 var props = s.props;
125
126
127 var padding = 16;
128
129 var w = props.width - padding * 2;
130 var h = props.height;
131
132 var moveRange = h / 20;
133 var move = _randomval2.default.randomInt.bind(_randomval2.default, moveRange * -1, moveRange);
134
135 var fontSize = h * 0.8;
136 var x = padding + w * rate + fontSize / 4;
137 var y = fontSize;
138 var rotateRange = 40;
139 var rotate = _randomval2.default.randomInt(-rotateRange, rotateRange);
140
141 return _react2.default.createElement(
142 'text',
143 _extends({ x: parseInt(x),
144 y: parseInt(y),
145 fontSize: parseInt(fontSize),
146 transform: 'translate(' + move() + ', ' + move() + ') rotate(' + parseInt(rotate) + ', ' + parseInt(x) + ', ' + parseInt(y) + ')'
147 }, textProps),
148 letter
149 );
150 }
151 }, {
152 key: '_dummyLetter',
153 value: function _dummyLetter() {
154 var letters = '1234567890abcdefg';
155 var len = letters.length;
156 return letters[_randomval2.default.randomInt(0, len - 1)];
157 }
158 }, {
159 key: '_stripeBlock',
160 value: function _stripeBlock(rate, color, blockProps) {
161 var s = this;
162 var props = s.props;
163
164
165 var rotate = _randomval2.default.randomInt(-90, 90);
166
167 var lines = [];
168 var lineWidth = 1;
169 var w = blockProps.width;
170 var h = blockProps.height;
171
172 var left = w * -0.5;
173 var right = w * 1.5;
174
175 for (var x = left; x < right; x += lineWidth * 6) {
176 lines.push(_react2.default.createElement('line', { x1: parseInt(x),
177 y1: parseInt(0),
178 x2: parseInt(x),
179 y2: parseInt(h),
180 key: 'line-' + x,
181 stroke: color
182 }));
183 }
184
185 return _react2.default.createElement(
186 'svg',
187 blockProps,
188 _react2.default.createElement(
189 'g',
190 {
191 transform: 'scale(1.3) rotate(' + parseInt(rotate) + ', ' + parseInt(blockProps.width / 2) + ', ' + parseInt(blockProps.height / 2) + ')' },
192 lines
193 )
194 );
195 }
196 }]);
197
198 return ApCaptchaSvg;
199}(_react.Component);
200
201Object.assign(ApCaptchaSvg, {
202 // --------------------
203 // Specs
204 // --------------------
205
206 propsTypes: {
207 version: _react.PropTypes.string,
208 text: _react.PropTypes.string.isRequired,
209 width: _react.PropTypes.number,
210 height: _react.PropTypes.number
211 },
212
213 defaultProps: {
214 version: '1.1',
215 xmlns: 'http://www.w3.org/2000/svg',
216 text: '',
217 width: 256,
218 height: 92
219 }
220});
221
222exports.default = ApCaptchaSvg;
223//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["ap_captcha_svg.jsx"],"names":[],"mappings":"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","file":"ap_captcha_svg.jsx","sourceRoot":"lib","sourcesContent":["/**\n * SVG captcha.\n * DO NOT use svg base captcha in production, since it may be parsable by bots.\n * You need to convert to bitmap like png beforehand.\n * @class ApCaptchaSvg\n */\n\n'use strict'\n\nimport React, {Component, PropTypes as types} from 'react'\nimport randomval from 'randomval'\n\n/** @lends ApCaptchaSvg */\nclass ApCaptchaSvg extends Component {\n  render () {\n    const s = this\n    let { props } = s\n\n    let color = '#555'\n\n    let { width, height } = props\n\n    let backgrounds = [ 0, 1, 2, 3 ].map((val, i, { length }) => {\n      let margin = width * 0.2\n      let rate = (i + 0.5) / length\n      return s._stripeBlock(rate, color, {\n        key: `bg-${i}`,\n        width: parseInt(width / length + (margin * 2)),\n        height: parseInt(height + (margin * 2)),\n        x: parseInt(width * rate - margin),\n        y: parseInt(0 - margin)\n      })\n    })\n\n    let texts = props.text.split('').map((letter, i, letters) => {\n      let indices = [ 0, 1, 2, 3, 4, 5 ]\n      let real = randomval.randomInt(0, indices.length - 1)\n      let texts = indices.map((j) => {\n        let rate = i / letters.length\n        let key = `letter-${i}-${j}`\n        if (j === real) {\n          return s.renderLetter(letter, rate, {\n            key: key,\n            fill: color\n          })\n        } else {\n          return s.renderLetter(s._dummyLetter(), rate, {\n            key: key,\n            fill: `rgba(255,255,255,${0.01 * randomval.randomInt(0, 30)})`\n          })\n        }\n      })\n      return (\n        <g key={`letter-group-${i}`}>{ texts }</g>\n      )\n    })\n    return (\n      <svg version={ props.version }\n           width={ width }\n           height={ height }\n           xmlns={ props.xmlns }\n           viewBox={`0 0 ${width} ${height}`}\n      >\n        <g>{backgrounds}</g>\n        <g>{texts}</g>\n      </svg>\n    )\n  }\n\n  // --------------------\n  // Specs\n  // --------------------\n\n  renderLetter (letter, rate, textProps) {\n    const s = this\n    let { props } = s\n\n    let padding = 16\n\n    let w = props.width - (padding * 2)\n    let h = props.height\n\n    let moveRange = h / 20\n    let move = randomval.randomInt.bind(randomval, moveRange * -1, moveRange)\n\n    let fontSize = h * 0.8\n    let x = padding + w * rate + (fontSize / 4)\n    let y = fontSize\n    let rotateRange = 40\n    let rotate = randomval.randomInt(-rotateRange, rotateRange)\n\n    return (\n      <text x={ parseInt(x) }\n            y={ parseInt(y) }\n            fontSize={ parseInt(fontSize) }\n            transform={ `translate(${move()}, ${move()}) rotate(${parseInt(rotate)}, ${parseInt(x)}, ${parseInt(y)})` }\n            {...textProps}\n      >{ letter }</text>\n    )\n  }\n\n  _dummyLetter () {\n    const letters = '1234567890abcdefg'\n    let len = letters.length\n    return letters[ randomval.randomInt(0, len - 1) ]\n  }\n\n  _stripeBlock (rate, color, blockProps) {\n    const s = this\n    let { props } = s\n\n    let rotate = randomval.randomInt(-90, 90)\n\n    let lines = []\n    let lineWidth = 1\n    let w = blockProps.width\n    let h = blockProps.height\n\n    let left = w * -0.5\n    let right = w * 1.5\n\n    for (let x = left; x < right; x += lineWidth * 6) {\n      lines.push(\n        <line x1={parseInt(x)}\n              y1={parseInt(0)}\n              x2={parseInt(x)}\n              y2={parseInt(h)}\n              key={`line-${x}`}\n              stroke={color}\n        />\n      )\n    }\n\n    return (\n      <svg { ...blockProps }>\n        <g\n          transform={`scale(1.3) rotate(${parseInt(rotate)}, ${parseInt(blockProps.width / 2)}, ${parseInt(blockProps.height / 2)})`}>\n          { lines }\n        </g>\n      </svg>\n    )\n  }\n}\nObject.assign(ApCaptchaSvg, {\n  // --------------------\n  // Specs\n  // --------------------\n\n  propsTypes: {\n    version: types.string,\n    text: types.string.isRequired,\n    width: types.number,\n    height: types.number\n  },\n\n  defaultProps: {\n    version: '1.1',\n    xmlns: 'http://www.w3.org/2000/svg',\n    text: '',\n    width: 256,\n    height: 92\n  }\n})\n\nexport default ApCaptchaSvg\n"]}
\No newline at end of file