mini-smooth-signature.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  1. 'use strict';
  2. Object.defineProperty(exports, '__esModule', {
  3. value: true
  4. });
  5. exports.default = void 0;
  6. function _classCallCheck(instance, Constructor) {
  7. if (!(instance instanceof Constructor)) {
  8. throw new TypeError('Cannot call a class as a function');
  9. }
  10. }
  11. var SmoothSignature = function SmoothSignature(_ctx, _options) {
  12. var _this = this;
  13. _classCallCheck(this, SmoothSignature);
  14. this.ctx = void 0;
  15. this.canvas = void 0;
  16. this.width = 300;
  17. this.height = 150;
  18. this.scale = 1;
  19. this.color = 'black';
  20. this.bgColor = '';
  21. this.openSmooth = true;
  22. this.minWidth = 2;
  23. this.maxWidth = 6;
  24. this.minSpeed = 1.5;
  25. this.maxWidthDiffRate = 20;
  26. this.maxHistoryLength = 20;
  27. this.points = [];
  28. this.canAddHistory = true;
  29. this.historyList = [];
  30. this.getImagePath = void 0;
  31. this.toDataURL = void 0;
  32. this.requestAnimationFrame = void 0;
  33. this.init = function (ctx, options) {
  34. if (!ctx) return;
  35. _this.ctx = ctx;
  36. _this.getImagePath = options.getImagePath;
  37. _this.toDataURL = options.toDataURL;
  38. _this.requestAnimationFrame = options.requestAnimationFrame;
  39. _this.width = options.width || _this.width;
  40. _this.height = options.height || _this.height;
  41. _this.color = options.color || _this.color;
  42. _this.bgColor = options.bgColor || _this.bgColor;
  43. _this.openSmooth = options.openSmooth || _this.openSmooth;
  44. _this.minWidth = options.minWidth || _this.minWidth;
  45. _this.maxWidth = options.maxWidth || _this.maxWidth;
  46. _this.minSpeed = options.minSpeed || _this.minSpeed;
  47. _this.maxWidthDiffRate = options.maxWidthDiffRate || _this.maxWidthDiffRate;
  48. _this.maxHistoryLength = options.maxHistoryLength || _this.maxHistoryLength;
  49. if (typeof options.scale === 'number') {
  50. _this.scale = options.scale;
  51. _this.ctx.scale(_this.scale, _this.scale);
  52. _this.ctx.draw && _this.ctx.draw();
  53. }
  54. _this.ctx.lineCap = 'round';
  55. _this.ctx.setLineCap && _this.ctx.setLineCap('round');
  56. _this.drawBgColor();
  57. };
  58. this.onDrawStart = function (x, y) {
  59. _this.canAddHistory = true;
  60. _this.ctx.strokeStyle = _this.color;
  61. _this.ctx.setStrokeStyle && _this.ctx.setStrokeStyle(_this.color);
  62. _this.initPoint(x, y);
  63. };
  64. this.onDrawMove = function (x, y) {
  65. _this.initPoint(x, y);
  66. _this.onDraw();
  67. };
  68. this.onDraw = function () {
  69. if (_this.points.length < 2) return;
  70. _this.addHistory();
  71. var point = _this.points.slice(-1)[0];
  72. var prePoint = _this.points.slice(-2, -1)[0];
  73. var onDraw = function onDraw() {
  74. if (_this.openSmooth) {
  75. _this.drawSmoothLine(prePoint, point);
  76. } else {
  77. _this.drawNoSmoothLine(prePoint, point);
  78. }
  79. };
  80. if (typeof _this.requestAnimationFrame === 'function') {
  81. _this.requestAnimationFrame(function () {
  82. return onDraw();
  83. });
  84. } else {
  85. onDraw();
  86. }
  87. };
  88. this.onDrawEnd = function () {
  89. _this.canAddHistory = true;
  90. _this.points = [];
  91. };
  92. this.getLineWidth = function (speed) {
  93. var minSpeed = _this.minSpeed > 10 ? 10 : _this.minSpeed < 1 ? 1 : _this.minSpeed;
  94. var addWidth = ((_this.maxWidth - _this.minWidth) * speed) / minSpeed;
  95. var lineWidth = Math.max(_this.maxWidth - addWidth, _this.minWidth);
  96. return Math.min(lineWidth, _this.maxWidth);
  97. };
  98. this.getRadianData = function (x1, y1, x2, y2) {
  99. var dis_x = x2 - x1;
  100. var dis_y = y2 - y1;
  101. if (dis_x === 0) {
  102. return {
  103. val: 0,
  104. pos: -1
  105. };
  106. }
  107. if (dis_y === 0) {
  108. return {
  109. val: 0,
  110. pos: 1
  111. };
  112. }
  113. var val = Math.abs(Math.atan(dis_y / dis_x));
  114. if ((x2 > x1 && y2 < y1) || (x2 < x1 && y2 > y1)) {
  115. return {
  116. val: val,
  117. pos: 1
  118. };
  119. }
  120. return {
  121. val: val,
  122. pos: -1
  123. };
  124. };
  125. this.getRadianPoints = function (radianData, x, y, halfLineWidth) {
  126. if (radianData.val === 0) {
  127. if (radianData.pos === 1) {
  128. return [
  129. {
  130. x: x,
  131. y: y + halfLineWidth
  132. },
  133. {
  134. x: x,
  135. y: y - halfLineWidth
  136. }
  137. ];
  138. }
  139. return [
  140. {
  141. y: y,
  142. x: x + halfLineWidth
  143. },
  144. {
  145. y: y,
  146. x: x - halfLineWidth
  147. }
  148. ];
  149. }
  150. var dis_x = Math.sin(radianData.val) * halfLineWidth;
  151. var dis_y = Math.cos(radianData.val) * halfLineWidth;
  152. if (radianData.pos === 1) {
  153. return [
  154. {
  155. x: x + dis_x,
  156. y: y + dis_y
  157. },
  158. {
  159. x: x - dis_x,
  160. y: y - dis_y
  161. }
  162. ];
  163. }
  164. return [
  165. {
  166. x: x + dis_x,
  167. y: y - dis_y
  168. },
  169. {
  170. x: x - dis_x,
  171. y: y + dis_y
  172. }
  173. ];
  174. };
  175. this.initPoint = function (x, y) {
  176. var point = {
  177. x: x,
  178. y: y,
  179. t: Date.now()
  180. };
  181. var prePoint = _this.points.slice(-1)[0];
  182. if (prePoint && (prePoint.t === point.t || (prePoint.x === x && prePoint.y === y))) {
  183. return;
  184. }
  185. if (_this.openSmooth && prePoint) {
  186. var prePoint2 = _this.points.slice(-2, -1)[0];
  187. point.distance = Math.sqrt(Math.pow(point.x - prePoint.x, 2) + Math.pow(point.y - prePoint.y, 2));
  188. point.speed = point.distance / (point.t - prePoint.t || 0.1);
  189. point.lineWidth = _this.getLineWidth(point.speed);
  190. if (prePoint2 && prePoint2.lineWidth && prePoint.lineWidth) {
  191. var rate = (point.lineWidth - prePoint.lineWidth) / prePoint.lineWidth;
  192. var maxRate = _this.maxWidthDiffRate / 100;
  193. maxRate = maxRate > 1 ? 1 : maxRate < 0.01 ? 0.01 : maxRate;
  194. if (Math.abs(rate) > maxRate) {
  195. var per = rate > 0 ? maxRate : -maxRate;
  196. point.lineWidth = prePoint.lineWidth * (1 + per);
  197. }
  198. }
  199. }
  200. _this.points.push(point);
  201. _this.points = _this.points.slice(-3);
  202. };
  203. this.drawSmoothLine = function (prePoint, point) {
  204. var dis_x = point.x - prePoint.x;
  205. var dis_y = point.y - prePoint.y;
  206. if (Math.abs(dis_x) + Math.abs(dis_y) <= 2) {
  207. point.lastX1 = point.lastX2 = prePoint.x + dis_x * 0.5;
  208. point.lastY1 = point.lastY2 = prePoint.y + dis_y * 0.5;
  209. } else {
  210. point.lastX1 = prePoint.x + dis_x * 0.3;
  211. point.lastY1 = prePoint.y + dis_y * 0.3;
  212. point.lastX2 = prePoint.x + dis_x * 0.7;
  213. point.lastY2 = prePoint.y + dis_y * 0.7;
  214. }
  215. point.perLineWidth = (prePoint.lineWidth + point.lineWidth) / 2;
  216. if (typeof prePoint.lastX1 === 'number') {
  217. _this.drawCurveLine(prePoint.lastX2, prePoint.lastY2, prePoint.x, prePoint.y, point.lastX1, point.lastY1, point.perLineWidth);
  218. if (prePoint.isFirstPoint) return;
  219. if (prePoint.lastX1 === prePoint.lastX2 && prePoint.lastY1 === prePoint.lastY2) return;
  220. var data = _this.getRadianData(prePoint.lastX1, prePoint.lastY1, prePoint.lastX2, prePoint.lastY2);
  221. var points1 = _this.getRadianPoints(data, prePoint.lastX1, prePoint.lastY1, prePoint.perLineWidth / 2);
  222. var points2 = _this.getRadianPoints(data, prePoint.lastX2, prePoint.lastY2, point.perLineWidth / 2);
  223. _this.drawTrapezoid(points1[0], points2[0], points2[1], points1[1]);
  224. } else {
  225. point.isFirstPoint = true;
  226. }
  227. };
  228. this.drawNoSmoothLine = function (prePoint, point) {
  229. point.lastX = prePoint.x + (point.x - prePoint.x) * 0.5;
  230. point.lastY = prePoint.y + (point.y - prePoint.y) * 0.5;
  231. if (typeof prePoint.lastX === 'number') {
  232. _this.drawCurveLine(prePoint.lastX, prePoint.lastY, prePoint.x, prePoint.y, point.lastX, point.lastY, _this.maxWidth);
  233. }
  234. };
  235. this.drawCurveLine = function (x1, y1, x2, y2, x3, y3, lineWidth) {
  236. lineWidth = Number(lineWidth.toFixed(1));
  237. _this.ctx.setLineWidth && _this.ctx.setLineWidth(lineWidth);
  238. _this.ctx.lineWidth = lineWidth;
  239. _this.ctx.beginPath();
  240. _this.ctx.moveTo(Number(x1.toFixed(1)), Number(y1.toFixed(1)));
  241. _this.ctx.quadraticCurveTo(Number(x2.toFixed(1)), Number(y2.toFixed(1)), Number(x3.toFixed(1)), Number(y3.toFixed(1)));
  242. _this.ctx.stroke();
  243. _this.ctx.draw && _this.ctx.draw(true);
  244. };
  245. this.drawTrapezoid = function (point1, point2, point3, point4) {
  246. _this.ctx.beginPath();
  247. _this.ctx.moveTo(Number(point1.x.toFixed(1)), Number(point1.y.toFixed(1)));
  248. _this.ctx.lineTo(Number(point2.x.toFixed(1)), Number(point2.y.toFixed(1)));
  249. _this.ctx.lineTo(Number(point3.x.toFixed(1)), Number(point3.y.toFixed(1)));
  250. _this.ctx.lineTo(Number(point4.x.toFixed(1)), Number(point4.y.toFixed(1)));
  251. _this.ctx.setFillStyle && _this.ctx.setFillStyle(_this.color);
  252. _this.ctx.fillStyle = _this.color;
  253. _this.ctx.fill();
  254. _this.ctx.draw && _this.ctx.draw(true);
  255. };
  256. this.drawBgColor = function () {
  257. if (!_this.bgColor) return;
  258. _this.ctx.setFillStyle && _this.ctx.setFillStyle(_this.bgColor);
  259. _this.ctx.fillStyle = _this.bgColor;
  260. _this.ctx.fillRect(0, 0, _this.width, _this.height);
  261. _this.ctx.draw && _this.ctx.draw(true);
  262. };
  263. this.drawByImage = function (url) {
  264. _this.ctx.clearRect(0, 0, _this.width, _this.height);
  265. try {
  266. _this.ctx.drawImage(url, 0, 0, _this.width, _this.height);
  267. _this.ctx.draw && _this.ctx.draw(true);
  268. } catch (e) {
  269. _this.historyList.length = 0;
  270. }
  271. };
  272. this.addHistory = function () {
  273. if (!_this.maxHistoryLength || !_this.canAddHistory) return;
  274. _this.canAddHistory = false;
  275. if (!_this.getImagePath) {
  276. _this.historyList.length++;
  277. return;
  278. }
  279. _this.getImagePath().then(function (url) {
  280. if (url) {
  281. _this.historyList.push(url);
  282. _this.historyList = _this.historyList.slice(-_this.maxHistoryLength);
  283. }
  284. });
  285. };
  286. this.clear = function () {
  287. _this.ctx.clearRect(0, 0, _this.width, _this.height);
  288. _this.ctx.draw && _this.ctx.draw();
  289. _this.drawBgColor();
  290. _this.historyList.length = 0;
  291. };
  292. this.undo = function () {
  293. if (!_this.getImagePath || !_this.historyList.length) return;
  294. var pngURL = _this.historyList.splice(-1)[0];
  295. _this.drawByImage(pngURL);
  296. if (_this.historyList.length === 0) {
  297. _this.clear();
  298. }
  299. };
  300. this.isEmpty = function () {
  301. return _this.historyList.length === 0;
  302. };
  303. this.init(_ctx, _options);
  304. };
  305. module.exports = SmoothSignature;
  306. var _default = SmoothSignature;
  307. exports.default = _default;