barcode.js 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  1. const barcodes = require('./barcodes/index.js')['default'];
  2. let barcode = {};
  3. (function () {
  4. // 初始化
  5. barcode = function (cont, ctxid, options, ctxsize, result) {
  6. let ops = {}, newOptions, encodings, globaContext, ctx, globaCtxid, cbCanvasSize, cbResult;
  7. globaCtxid = ctxid
  8. cbCanvasSize = ctxsize
  9. cbResult = result
  10. newOptions = Object.assign(ops, options);
  11. // 修成margin
  12. fixMargin(newOptions)
  13. // 处理options 数据
  14. if (newOptions.text == '' || cont == '') {
  15. return false
  16. }
  17. // 获取ctx
  18. globaContext = cont
  19. ctx = uni.createCanvasContext(globaCtxid, globaContext)
  20. // 获取编码数据
  21. encodings = new barcodes[newOptions.format.toUpperCase()](newOptions.text, newOptions).encode()
  22. let fixencodings = fixEncodings(encodings, newOptions)
  23. // 返回canvas实际大小
  24. cbCanvasSize({ width: fixencodings.width, height: fixencodings.height })
  25. // 绘制canvas
  26. setTimeout(() => {
  27. drawCanvas.render(newOptions, fixencodings)
  28. }, 50);
  29. // 绘制canvas
  30. let drawCanvas = {
  31. render(options, encoding) {
  32. this.prepare(options, encoding)
  33. encoding.encodings.forEach((v, i) => {
  34. this.barcode(options, v)
  35. this.text(options, v)
  36. this.move(v)
  37. });
  38. this.draw(options, encoding)
  39. },
  40. barcode(options, encoding) {
  41. let binary = encoding.data;
  42. let yFrom;
  43. if (options.textPosition == "top") {
  44. yFrom = options.marginTop + options.fontSize + options.textMargin;
  45. } else {
  46. yFrom = options.marginTop;
  47. }
  48. // 绘制条码
  49. ctx.fillStyle = options.lineColor;
  50. for (let b = 0; b < binary.length; b++) {
  51. let x = b * options.width + encoding.barcodePadding;
  52. let height = options.height
  53. if (encoding.options) {
  54. if (encoding.options.height != undefined) {
  55. height = encoding.options.height
  56. }
  57. }
  58. if (binary[b] === "1") {
  59. ctx.fillRect(x, yFrom, options.width, height);
  60. } else if (binary[b]) {
  61. ctx.fillRect(x, yFrom, options.width, height * binary[b]);
  62. }
  63. }
  64. },
  65. text(options, encoding) {
  66. if (options.displayValue) {
  67. let x, y, align, size;
  68. if (options.textPosition == "top") {
  69. y = options.marginTop + options.fontSize;
  70. } else {
  71. y = options.height + options.textMargin + options.marginTop + options.fontSize;
  72. }
  73. if (encoding.options) {
  74. if (encoding.options.textAlign != undefined) {
  75. align = encoding.options.textAlign
  76. }
  77. if (encoding.options.fontSize != undefined) {
  78. size = encoding.options.fontSize
  79. }
  80. } else {
  81. align = options.textAlign
  82. size = options.fontSize
  83. }
  84. ctx.setFontSize(size)
  85. if (align == "left" || encoding.barcodePadding > 0) {
  86. x = 0;
  87. ctx.setTextAlign('left')
  88. } else if (align == "right") {
  89. x = encoding.width - 1;
  90. ctx.setTextAlign('right')
  91. }
  92. else {
  93. x = encoding.width / 2;
  94. ctx.setTextAlign('center');
  95. }
  96. ctx.fillStyle = options.fontColor;
  97. if (encoding.text != undefined) {
  98. ctx.fillText(encoding.text, x, y);
  99. }
  100. }
  101. },
  102. move(encoding) {
  103. ctx.translate(encoding.width, 0);
  104. },
  105. prepare(options, encoding) {
  106. // 绘制背景
  107. if (options.background) {
  108. ctx.fillStyle = options.background;
  109. ctx.fillRect(0, 0, encoding.width, encoding.height);
  110. }
  111. ctx.translate(options.marginLeft, 0);
  112. },
  113. draw(options, encoding) {
  114. ctx.draw(false, () => {
  115. this.toImgs(options, encoding)
  116. })
  117. },
  118. toImgs(options, encoding) {
  119. setTimeout(() => {
  120. uni.canvasToTempFilePath({
  121. width: encoding.width,
  122. height: encoding.height,
  123. destWidth: encoding.width,
  124. destHeight: encoding.height,
  125. canvasId: globaCtxid,
  126. fileType: 'png',
  127. success: function (res) {
  128. cbResult(res.tempFilePath)
  129. },
  130. fail: function (res) {
  131. cbResult(res)
  132. },
  133. complete: function () {
  134. uni.hideLoading();
  135. },
  136. }, globaContext);
  137. }, options.text.length + 100);
  138. }
  139. }
  140. // 混入canvas数据
  141. function fixEncodings(encoding, options) {
  142. let encodingArr = [], width = options.marginLeft + options.marginRight, height;
  143. if (!Array.isArray(encoding)) {
  144. encodingArr[0] = JSON.parse(JSON.stringify(encoding))
  145. } else {
  146. encodingArr = [...encoding]
  147. }
  148. encodingArr.forEach((v, i) => {
  149. // 获取文本宽度
  150. let textWidth = ctx.measureText(encodingArr[i].text ? encodingArr[i].text : '').width;
  151. // 获取条形码宽度
  152. let barcodeWidth = encodingArr[i].data.length * options.width;
  153. // 获取内边距
  154. let barcodePadding = 0;
  155. if (options.displayValue && barcodeWidth < textWidth) {
  156. if (options.textAlign == "center") {
  157. barcodePadding = Math.floor((textWidth - barcodeWidth) / 2);
  158. } else if (options.textAlign == "left") {
  159. barcodePadding = 0;
  160. } else if (options.textAlign == "right") {
  161. barcodePadding = Math.floor(textWidth - barcodeWidth);
  162. }
  163. }
  164. // 混入encodingArr[i]
  165. encodingArr[i].barcodePadding = barcodePadding
  166. encodingArr[i].width = Math.ceil(Math.max(textWidth, barcodeWidth))
  167. width += encodingArr[i].width
  168. if (encodingArr[i].options) {
  169. if (encodingArr[i].options.height != undefined) {
  170. encodingArr[i].height = encodingArr[i].options.height + (options.displayValue && (encodingArr[i].text ? encodingArr[i].text : '').length > 0 ? options.fontSize + options.textMargin : 0) + options.marginTop + options.marginBottom;
  171. } else {
  172. encodingArr[i].height = height = options.height + (options.displayValue && (encodingArr[i].text ? encodingArr[i].text : '').length > 0 ? options.fontSize + options.textMargin : 0) + options.marginTop + options.marginBottom;
  173. }
  174. } else {
  175. encodingArr[i].height = height = options.height + (options.displayValue && (encodingArr[i].text ? encodingArr[i].text : '').length > 0 ? options.fontSize + options.textMargin : 0) + options.marginTop + options.marginBottom;
  176. }
  177. });
  178. return { encodings: encodingArr, width, height };
  179. }
  180. // 修正Margin
  181. function fixMargin(options) {
  182. options.marginTop = options.marginTop == undefined ? options.margin : options.marginTop;
  183. options.marginBottom = options.marginBottom == undefined ? options.margin : options.marginBottom;
  184. options.marginRight = options.marginRight == undefined ? options.margin : options.marginRight;
  185. options.marginLeft = options.marginLeft == undefined ? options.margin : options.marginLeft;
  186. }
  187. };
  188. })()
  189. export default barcode