/* * Copyright 2012 ZXing authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #import "ZXBitArray.h" #import "ZXCode39Reader.h" #import "ZXErrors.h" #import "ZXIntArray.h" #import "ZXResult.h" #import "ZXResultPoint.h" unichar ZX_CODE39_ALPHABET[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', '-', '.', ' ', '*', '$', '/', '+', '%'}; NSString *ZX_CODE39_ALPHABET_STRING = nil; /** * These represent the encodings of characters, as patterns of wide and narrow bars. * The 9 least-significant bits of each int correspond to the pattern of wide and narrow, * with 1s representing "wide" and 0s representing narrow. */ const int ZX_CODE39_CHARACTER_ENCODINGS[] = { 0x034, 0x121, 0x061, 0x160, 0x031, 0x130, 0x070, 0x025, 0x124, 0x064, // 0-9 0x109, 0x049, 0x148, 0x019, 0x118, 0x058, 0x00D, 0x10C, 0x04C, 0x01C, // A-J 0x103, 0x043, 0x142, 0x013, 0x112, 0x052, 0x007, 0x106, 0x046, 0x016, // K-T 0x181, 0x0C1, 0x1C0, 0x091, 0x190, 0x0D0, 0x085, 0x184, 0x0C4, 0x094, // U-* 0x0A8, 0x0A2, 0x08A, 0x02A // $-% }; const int ZX_CODE39_ASTERISK_ENCODING = 0x094; @interface ZXCode39Reader () @property (nonatomic, assign, readonly) BOOL extendedMode; @property (nonatomic, assign, readonly) BOOL usingCheckDigit; @property (nonatomic, strong, readonly) ZXIntArray *counters; @end @implementation ZXCode39Reader + (void)load { ZX_CODE39_ALPHABET_STRING = [[NSString alloc] initWithCharacters:ZX_CODE39_ALPHABET length:sizeof(ZX_CODE39_ALPHABET) / sizeof(unichar)]; } - (id)init { return [self initUsingCheckDigit:NO extendedMode:NO]; } - (id)initUsingCheckDigit:(BOOL)isUsingCheckDigit { return [self initUsingCheckDigit:isUsingCheckDigit extendedMode:NO]; } - (id)initUsingCheckDigit:(BOOL)usingCheckDigit extendedMode:(BOOL)extendedMode { if (self = [super init]) { _usingCheckDigit = usingCheckDigit; _extendedMode = extendedMode; _counters = [[ZXIntArray alloc] initWithLength:9]; } return self; } - (ZXResult *)decodeRow:(int)rowNumber row:(ZXBitArray *)row hints:(ZXDecodeHints *)hints error:(NSError **)error { ZXIntArray *theCounters = self.counters; [theCounters clear]; NSMutableString *result = [NSMutableString stringWithCapacity:20]; ZXIntArray *start = [self findAsteriskPattern:row counters:theCounters]; if (!start) { if (error) *error = ZXNotFoundErrorInstance(); return nil; } // Read off white space int nextStart = [row nextSet:start.array[1]]; int end = [row size]; unichar decodedChar; int lastStart; do { if (![ZXOneDReader recordPattern:row start:nextStart counters:theCounters]) { if (error) *error = ZXNotFoundErrorInstance(); return nil; } int pattern = [self toNarrowWidePattern:theCounters]; if (pattern < 0) { if (error) *error = ZXNotFoundErrorInstance(); return nil; } decodedChar = [self patternToChar:pattern]; if (decodedChar == 0) { if (error) *error = ZXNotFoundErrorInstance(); return nil; } [result appendFormat:@"%C", decodedChar]; lastStart = nextStart; for (int i = 0; i < theCounters.length; i++) { nextStart += theCounters.array[i]; } // Read off white space nextStart = [row nextSet:nextStart]; } while (decodedChar != '*'); [result deleteCharactersInRange:NSMakeRange([result length] - 1, 1)]; int lastPatternSize = 0; for (int i = 0; i < theCounters.length; i++) { lastPatternSize += theCounters.array[i]; } int whiteSpaceAfterEnd = nextStart - lastStart - lastPatternSize; if (nextStart != end && (whiteSpaceAfterEnd * 2) < lastPatternSize) { if (error) *error = ZXNotFoundErrorInstance(); return nil; } if (self.usingCheckDigit) { int max = (int)[result length] - 1; int total = 0; for (int i = 0; i < max; i++) { total += [ZX_CODE39_ALPHABET_STRING rangeOfString:[result substringWithRange:NSMakeRange(i, 1)]].location; } if ([result characterAtIndex:max] != ZX_CODE39_ALPHABET[total % 43]) { if (error) *error = ZXChecksumErrorInstance(); return nil; } [result deleteCharactersInRange:NSMakeRange(max, 1)]; } if ([result length] == 0) { // false positive if (error) *error = ZXNotFoundErrorInstance(); return nil; } NSString *resultString; if (self.extendedMode) { resultString = [self decodeExtended:result]; if (!resultString) { if (error) *error = ZXFormatErrorInstance(); return nil; } } else { resultString = result; } float left = (float) (start.array[1] + start.array[0]) / 2.0f; float right = (lastStart + lastPatternSize) / 2.0f; return [ZXResult resultWithText:resultString rawBytes:nil resultPoints:@[[[ZXResultPoint alloc] initWithX:left y:(float)rowNumber], [[ZXResultPoint alloc] initWithX:right y:(float)rowNumber]] format:kBarcodeFormatCode39]; } - (ZXIntArray *)findAsteriskPattern:(ZXBitArray *)row counters:(ZXIntArray *)counters { int width = row.size; int rowOffset = [row nextSet:0]; int counterPosition = 0; int patternStart = rowOffset; BOOL isWhite = NO; int patternLength = counters.length; int32_t *array = counters.array; for (int i = rowOffset; i < width; i++) { if ([row get:i] ^ isWhite) { array[counterPosition]++; } else { if (counterPosition == patternLength - 1) { if ([self toNarrowWidePattern:counters] == ZX_CODE39_ASTERISK_ENCODING && [row isRange:MAX(0, patternStart - ((i - patternStart) / 2)) end:patternStart value:NO]) { return [[ZXIntArray alloc] initWithInts:patternLength, i, -1]; } patternStart += array[0] + array[1]; for (int y = 2; y < counters.length; y++) { array[y - 2] = array[y]; } array[patternLength - 2] = 0; array[patternLength - 1] = 0; counterPosition--; } else { counterPosition++; } array[counterPosition] = 1; isWhite = !isWhite; } } return nil; } - (int)toNarrowWidePattern:(ZXIntArray *)counters { int numCounters = counters.length; int maxNarrowCounter = 0; int wideCounters; do { int minCounter = INT_MAX; int32_t *array = counters.array; for (int i = 0; i < numCounters; i++) { int counter = array[i]; if (counter < minCounter && counter > maxNarrowCounter) { minCounter = counter; } } maxNarrowCounter = minCounter; wideCounters = 0; int totalWideCountersWidth = 0; int pattern = 0; for (int i = 0; i < numCounters; i++) { int counter = array[i]; if (array[i] > maxNarrowCounter) { pattern |= 1 << (numCounters - 1 - i); wideCounters++; totalWideCountersWidth += counter; } } if (wideCounters == 3) { for (int i = 0; i < numCounters && wideCounters > 0; i++) { int counter = array[i]; if (array[i] > maxNarrowCounter) { wideCounters--; if ((counter * 2) >= totalWideCountersWidth) { return -1; } } } return pattern; } } while (wideCounters > 3); return -1; } - (unichar)patternToChar:(int)pattern { for (int i = 0; i < sizeof(ZX_CODE39_CHARACTER_ENCODINGS) / sizeof(int); i++) { if (ZX_CODE39_CHARACTER_ENCODINGS[i] == pattern) { return ZX_CODE39_ALPHABET[i]; } } return 0; } - (NSString *)decodeExtended:(NSMutableString *)encoded { NSUInteger length = [encoded length]; NSMutableString *decoded = [NSMutableString stringWithCapacity:length]; for (int i = 0; i < length; i++) { unichar c = [encoded characterAtIndex:i]; if (c == '+' || c == '$' || c == '%' || c == '/') { unichar next = [encoded characterAtIndex:i + 1]; unichar decodedChar = '\0'; switch (c) { case '+': if (next >= 'A' && next <= 'Z') { decodedChar = (unichar)(next + 32); } else { return nil; } break; case '$': if (next >= 'A' && next <= 'Z') { decodedChar = (unichar)(next - 64); } else { return nil; } break; case '%': if (next >= 'A' && next <= 'E') { decodedChar = (unichar)(next - 38); } else if (next >= 'F' && next <= 'W') { decodedChar = (unichar)(next - 11); } else { return nil; } break; case '/': if (next >= 'A' && next <= 'O') { decodedChar = (unichar)(next - 32); } else if (next == 'Z') { decodedChar = ':'; } else { return nil; } break; } [decoded appendFormat:@"%C", decodedChar]; i++; } else { [decoded appendFormat:@"%C", c]; } } return decoded; } @end