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 | ;
|
9 |
|
10 | Object.defineProperty(exports, "__esModule", {
|
11 | value: true
|
12 | });
|
13 |
|
14 | var _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 |
|
16 | var _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 |
|
18 | var _react = require('react');
|
19 |
|
20 | var _react2 = _interopRequireDefault(_react);
|
21 |
|
22 | var _randomval = require('randomval');
|
23 |
|
24 | var _randomval2 = _interopRequireDefault(_randomval);
|
25 |
|
26 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
27 |
|
28 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
|
29 |
|
30 | function _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 |
|
32 | function _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 */
|
35 | var 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 |
|
201 | Object.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 |
|
222 | exports.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 |