123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112 |
- /*
- * 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 "ZXDecodeHints.h"
- #import "ZXEAN8Reader.h"
- #import "ZXEAN13Reader.h"
- #import "ZXErrors.h"
- #import "ZXMultiFormatUPCEANReader.h"
- #import "ZXReader.h"
- #import "ZXResult.h"
- #import "ZXUPCAReader.h"
- #import "ZXUPCEReader.h"
-
- @interface ZXMultiFormatUPCEANReader ()
-
- @property (nonatomic, strong, readonly) NSMutableArray *readers;
-
- @end
-
- @implementation ZXMultiFormatUPCEANReader
-
- - (id)initWithHints:(ZXDecodeHints *)hints {
- if (self = [super init]) {
- _readers = [NSMutableArray array];
-
- if (hints != nil) {
- if ([hints containsFormat:kBarcodeFormatEan13]) {
- [_readers addObject:[[ZXEAN13Reader alloc] init]];
- } else if ([hints containsFormat:kBarcodeFormatUPCA]) {
- [_readers addObject:[[ZXUPCAReader alloc] init]];
- }
-
- if ([hints containsFormat:kBarcodeFormatEan8]) {
- [_readers addObject:[[ZXEAN8Reader alloc] init]];
- }
-
- if ([hints containsFormat:kBarcodeFormatUPCE]) {
- [_readers addObject:[[ZXUPCEReader alloc] init]];
- }
- }
-
- if ([_readers count] == 0) {
- [_readers addObject:[[ZXEAN13Reader alloc] init]];
- [_readers addObject:[[ZXEAN8Reader alloc] init]];
- [_readers addObject:[[ZXUPCEReader alloc] init]];
- }
- }
-
- return self;
- }
-
- - (ZXResult *)decodeRow:(int)rowNumber row:(ZXBitArray *)row hints:(ZXDecodeHints *)hints error:(NSError **)error {
- NSRange startGuardPattern = [ZXUPCEANReader findStartGuardPattern:row error:error];
- if (startGuardPattern.location == NSNotFound) {
- return nil;
- }
- for (ZXUPCEANReader *reader in self.readers) {
- ZXResult *result = [reader decodeRow:rowNumber row:row startGuardRange:startGuardPattern hints:hints error:error];
- if (!result) {
- continue;
- }
-
- // Special case: a 12-digit code encoded in UPC-A is identical to a "0"
- // followed by those 12 digits encoded as EAN-13. Each will recognize such a code,
- // UPC-A as a 12-digit string and EAN-13 as a 13-digit string starting with "0".
- // Individually these are correct and their readers will both read such a code
- // and correctly call it EAN-13, or UPC-A, respectively.
- //
- // In this case, if we've been looking for both types, we'd like to call it
- // a UPC-A code. But for efficiency we only run the EAN-13 decoder to also read
- // UPC-A. So we special case it here, and convert an EAN-13 result to a UPC-A
- // result if appropriate.
- //
- // But, don't return UPC-A if UPC-A was not a requested format!
- BOOL ean13MayBeUPCA = kBarcodeFormatEan13 == result.barcodeFormat && [result.text characterAtIndex:0] == '0';
- BOOL canReturnUPCA = hints == nil || [hints numberOfPossibleFormats] == 0 || [hints containsFormat:kBarcodeFormatUPCA];
- if (ean13MayBeUPCA && canReturnUPCA) {
- // Transfer the metdata across
- ZXResult *resultUPCA = [ZXResult resultWithText:[result.text substringFromIndex:1]
- rawBytes:result.rawBytes
- resultPoints:result.resultPoints
- format:kBarcodeFormatUPCA];
- [resultUPCA putAllMetadata:result.resultMetadata];
- return resultUPCA;
- }
- return result;
- }
-
- if (error) *error = ZXNotFoundErrorInstance();
- return nil;
- }
-
- - (void)reset {
- for (id<ZXReader> reader in self.readers) {
- [reader reset];
- }
- }
-
- @end
|