UNPKG

quagga

Version:

An advanced barcode-scanner written in JavaScript

252 lines (225 loc) 7.91 kB
import BarcodeReader from './barcode_reader'; import ArrayHelper from '../common/array_helper'; function Code93Reader() { BarcodeReader.call(this); } const ALPHABETH_STRING = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ-. $/+%abcd*"; var properties = { ALPHABETH_STRING: {value: ALPHABETH_STRING}, ALPHABET: {value: ALPHABETH_STRING.split('').map(char => char.charCodeAt(0))}, CHARACTER_ENCODINGS: {value: [ 0x114, 0x148, 0x144, 0x142, 0x128, 0x124, 0x122, 0x150, 0x112, 0x10A, 0x1A8, 0x1A4, 0x1A2, 0x194, 0x192, 0x18A, 0x168, 0x164, 0x162, 0x134, 0x11A, 0x158, 0x14C, 0x146, 0x12C, 0x116, 0x1B4, 0x1B2, 0x1AC, 0x1A6, 0x196, 0x19A, 0x16C, 0x166, 0x136, 0x13A, 0x12E, 0x1D4, 0x1D2, 0x1CA, 0x16E, 0x176, 0x1AE, 0x126, 0x1DA, 0x1D6, 0x132, 0x15E ]}, ASTERISK: {value: 0x15E}, FORMAT: {value: "code_93", writeable: false} }; Code93Reader.prototype = Object.create(BarcodeReader.prototype, properties); Code93Reader.prototype.constructor = Code93Reader; Code93Reader.prototype._decode = function() { var self = this, counters = [0, 0, 0, 0, 0, 0], result = [], start = self._findStart(), decodedChar, lastStart, pattern, nextStart; if (!start) { return null; } nextStart = self._nextSet(self._row, start.end); do { counters = self._toCounters(nextStart, counters); pattern = self._toPattern(counters); if (pattern < 0) { return null; } decodedChar = self._patternToChar(pattern); if (decodedChar < 0){ return null; } result.push(decodedChar); lastStart = nextStart; nextStart += ArrayHelper.sum(counters); nextStart = self._nextSet(self._row, nextStart); } while (decodedChar !== '*'); result.pop(); if (!result.length) { return null; } if (!self._verifyEnd(lastStart, nextStart, counters)) { return null; } if (!self._verifyChecksums(result)) { return null; } result = result.slice(0, result.length - 2); if ((result = self._decodeExtended(result)) === null) { return null; }; return { code: result.join(""), start: start.start, end: nextStart, startInfo: start, decodedCodes: result }; }; Code93Reader.prototype._verifyEnd = function(lastStart, nextStart) { if (lastStart === nextStart || !this._row[nextStart]) { return false; } return true; }; Code93Reader.prototype._patternToChar = function(pattern) { var i, self = this; for (i = 0; i < self.CHARACTER_ENCODINGS.length; i++) { if (self.CHARACTER_ENCODINGS[i] === pattern) { return String.fromCharCode(self.ALPHABET[i]); } } return -1; }; Code93Reader.prototype._toPattern = function(counters) { const numCounters = counters.length; let pattern = 0; let sum = 0; for (let i = 0; i < numCounters; i++) { sum += counters[i]; } for (let i = 0; i < numCounters; i++) { let normalized = Math.round(counters[i] * 9 / sum); if (normalized < 1 || normalized > 4) { return -1; } if ((i & 1) === 0) { for (let j = 0; j < normalized; j++) { pattern = (pattern << 1) | 1; } } else { pattern <<= normalized; } } return pattern; }; Code93Reader.prototype._findStart = function() { var self = this, offset = self._nextSet(self._row), patternStart = offset, counter = [0, 0, 0, 0, 0, 0], counterPos = 0, isWhite = false, i, j, whiteSpaceMustStart; for ( i = offset; i < self._row.length; i++) { if (self._row[i] ^ isWhite) { counter[counterPos]++; } else { if (counterPos === counter.length - 1) { // find start pattern if (self._toPattern(counter) === self.ASTERISK) { whiteSpaceMustStart = Math.floor(Math.max(0, patternStart - ((i - patternStart) / 4))); if (self._matchRange(whiteSpaceMustStart, patternStart, 0)) { return { start: patternStart, end: i }; } } patternStart += counter[0] + counter[1]; for ( j = 0; j < 4; j++) { counter[j] = counter[j + 2]; } counter[4] = 0; counter[5] = 0; counterPos--; } else { counterPos++; } counter[counterPos] = 1; isWhite = !isWhite; } } return null; }; Code93Reader.prototype._decodeExtended = function(charArray) { const length = charArray.length; const result = []; for (let i = 0; i < length; i++) { const char = charArray[i]; if (char >= 'a' && char <= 'd') { if (i > (length - 2)) { return null; } const nextChar = charArray[++i]; const nextCharCode = nextChar.charCodeAt(0); let decodedChar; switch (char) { case 'a': if (nextChar >= 'A' && nextChar <= 'Z') { decodedChar = String.fromCharCode(nextCharCode - 64); } else { return null; } break; case 'b': if (nextChar >= 'A' && nextChar <= 'E') { decodedChar = String.fromCharCode(nextCharCode - 38); } else if (nextChar >= 'F' && nextChar <= 'J') { decodedChar = String.fromCharCode(nextCharCode - 11); } else if (nextChar >= 'K' && nextChar <= 'O') { decodedChar = String.fromCharCode(nextCharCode + 16); } else if (nextChar >= 'P' && nextChar <= 'S') { decodedChar = String.fromCharCode(nextCharCode + 43); } else if (nextChar >= 'T' && nextChar <= 'Z') { decodedChar = String.fromCharCode(127); } else { return null; } break; case 'c': if (nextChar >= 'A' && nextChar <= 'O') { decodedChar = String.fromCharCode(nextCharCode - 32); } else if (nextChar === 'Z') { decodedChar = ':'; } else { return null; } break; case 'd': if (nextChar >= 'A' && nextChar <= 'Z') { decodedChar = String.fromCharCode(nextCharCode + 32); } else { return null; } break; } result.push(decodedChar); } else { result.push(char); } } return result; }; Code93Reader.prototype._verifyChecksums = function(charArray) { return this._matchCheckChar(charArray, charArray.length - 2, 20) && this._matchCheckChar(charArray, charArray.length - 1, 15); }; Code93Reader.prototype._matchCheckChar = function(charArray, index, maxWeight) { const arrayToCheck = charArray.slice(0, index); const length = arrayToCheck.length; const weightedSums = arrayToCheck.reduce((sum, char, i) => { const weight = (((i * -1) + (length - 1)) % maxWeight) + 1; const value = this.ALPHABET.indexOf(char.charCodeAt(0)); return sum + (weight * value); }, 0); const checkChar = this.ALPHABET[(weightedSums % 47)]; return checkChar === charArray[index].charCodeAt(0); }; export default Code93Reader;