diff options
Diffstat (limited to 'lib/DDHidKeyboardBarcodeScanner.m')
| -rw-r--r-- | lib/DDHidKeyboardBarcodeScanner.m | 222 | 
1 files changed, 222 insertions, 0 deletions
| diff --git a/lib/DDHidKeyboardBarcodeScanner.m b/lib/DDHidKeyboardBarcodeScanner.m new file mode 100644 index 0000000..8705cf3 --- /dev/null +++ b/lib/DDHidKeyboardBarcodeScanner.m @@ -0,0 +1,222 @@ +/* + * Copyright (c) 2007 Dave Dribin, Lucas Newman + *  + * Permission is hereby granted, free of charge, to any person + * obtaining a copy of this software and associated documentation + * files (the "Software"), to deal in the Software without + * restriction, including without limitation the rights to use, copy, + * modify, merge, publish, distribute, sublicense, and/or sell copies + * of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + *  + * The above copyright notice and this permission notice shall be + * included in all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS + * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN + * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + * SOFTWARE. + */ + +#import "DDHidKeyboardBarcodeScanner.h" +#import "DDHidElement.h" +#import "DDHidUsage.h" +#import "DDHidQueue.h" +#import "DDHidEvent.h" +#include <IOKit/hid/IOHIDUsageTables.h> + +@interface DDHidKeyboardBarcodeScanner (DDHidKeyboardBarcodeDelegate) + +- (void) ddhidKeyboardBarcodeScanner: (DDHidKeyboardBarcodeScanner *) keyboardBarcodeScanner +                          gotBarcode: (NSString *) barcode; + +@end + +@interface DDHidKeyboardBarcodeScanner (Private) + +- (void) initKeyboardElements: (NSArray *) elements; +- (void) ddhidQueueHasEvents: (DDHidQueue *) hidQueue; +- (void) processBarcodeDigit: (unsigned) usageId; +- (void) clearAccumulatedInput; +- (void) invalidateBarcodeInputTimer; + +@end + +@implementation DDHidKeyboardBarcodeScanner + ++ (NSArray *) allPossibleKeyboardBarcodeScanners; +{ +    return [DDHidDevice allDevicesMatchingUsagePage: kHIDPage_GenericDesktop +                                            usageId: kHIDUsage_GD_Keyboard +                                          withClass: self +                                  skipZeroLocations: YES]; +} + +- (id) initWithDevice: (io_object_t) device error: (NSError **) error_; +{ +    self = [super initWithDevice: device error: error_]; +    if (self == nil) +        return nil; +     +    mKeyElements = [[NSMutableArray alloc] init]; +    mAccumulatedDigits = [[NSMutableString alloc] init]; +    mBarcodeInputTimer = nil; +     +    if ([[self productName] rangeOfString:@"Apple"].location != NSNotFound || [[self productName] rangeOfString:@"Internal"].location != NSNotFound) +        mIsLikelyKeyboardBarcodeScanner = NO; +    else +        mIsLikelyKeyboardBarcodeScanner = YES; // if we see invalid barcodes, we can change our mind +     +    [self initKeyboardElements: [self elements]]; +     +    return self; +} + +//===========================================================  +// dealloc +//===========================================================  +- (void) dealloc +{ +    [self invalidateBarcodeInputTimer]; +    [mKeyElements release]; +    [mAccumulatedDigits release]; +     +    mKeyElements = nil; +    mAccumulatedDigits = nil; +    [super dealloc]; +} + +#pragma mark - +#pragma mark Keyboard Elements + +- (NSArray *) keyElements; +{ +    return mKeyElements; +} + +- (unsigned) numberOfKeys; +{ +    return [mKeyElements count]; +} + +- (void) addElementsToQueue: (DDHidQueue *) queue; +{ +    [queue addElements: mKeyElements]; +} + +#pragma mark - +#pragma mark Asynchronous Notification + +- (void) setDelegate: (id) delegate; +{ +    mDelegate = delegate; +} + +- (void) addElementsToDefaultQueue; +{ +    [self addElementsToQueue: mDefaultQueue]; +} + +#pragma mark - +#pragma mark Properties + +- (BOOL) isLikelyKeyboardBarcodeScanner; +{ +    return mIsLikelyKeyboardBarcodeScanner; +} + +@end + +@implementation DDHidKeyboardBarcodeScanner (DDHidKeyboardDelegate) + +- (void) ddhidKeyboardBarcodeScanner: (DDHidKeyboardBarcodeScanner *) keyboardBarcodeScanner +                          gotBarcode: (NSString *) barcode; +{ +    if ([mDelegate respondsToSelector: _cmd]) +        [mDelegate ddhidKeyboardBarcodeScanner: keyboardBarcodeScanner gotBarcode: barcode]; +} + +@end + +@implementation DDHidKeyboardBarcodeScanner (Private) + +- (void) initKeyboardElements: (NSArray *) elements; +{ +    NSEnumerator * e = [elements objectEnumerator]; +    DDHidElement * element; +    while ((element = [e nextObject])) +    { +        unsigned usagePage = [[element usage] usagePage]; +        unsigned usageId = [[element usage] usageId]; +        if (usagePage == kHIDPage_KeyboardOrKeypad) +        { +            if ((usageId >= kHIDUsage_KeyboardA) && (usageId <= kHIDUsage_Keyboard0)) +            { +                [mKeyElements addObject: element]; +            } +        } +        NSArray * subElements = [element elements]; +        if (subElements != nil) +            [self initKeyboardElements: subElements]; +    } +} + +- (void) ddhidQueueHasEvents: (DDHidQueue *) hidQueue; +{ +    DDHidEvent * event; +    while ((event = [hidQueue nextEvent])) +    { +        DDHidElement * element = [self elementForCookie: [event elementCookie]]; +        unsigned usageId = [[element usage] usageId]; +        SInt32 value = [event value]; +        if (value == 1) // key down +            [self processBarcodeDigit: usageId]; +    } +} + +#define UPC_A_BARCODE_LENGTH (12) +#define BARCODE_INPUT_TIMEOUT (0.5) +    +- (void) processBarcodeDigit: (unsigned) usageId; +{ +    if (usageId <= kHIDUsage_KeyboardZ || usageId >= kHIDUsage_KeyboardCapsLock) { // an alphabetic key was pressed => probably not a barcode scanner +        [self willChangeValueForKey:@"isLikelyKeyboardBarcodeScanner"]; +        mIsLikelyKeyboardBarcodeScanner = NO; +        [self didChangeValueForKey:@"isLikelyKeyboardBarcodeScanner"]; +         +        [self clearAccumulatedInput]; +        return; +    } +     +    if (!mBarcodeInputTimer) // schedule a timer to make sure we get the rest of the digits in a timely manner +        mBarcodeInputTimer = [[NSTimer scheduledTimerWithTimeInterval:BARCODE_INPUT_TIMEOUT target:self selector:@selector(fireBarcodeInputTimeout:) userInfo:nil repeats:NO] retain]; +     +    [mAccumulatedDigits appendString:[NSString stringWithFormat:@"%d", (usageId + 1) % 10]]; +} + +- (void) fireBarcodeInputTimeout: (NSTimer *) timer; +{ +    if ([mAccumulatedDigits length] >= UPC_A_BARCODE_LENGTH) +        [self ddhidKeyboardBarcodeScanner: self gotBarcode: [[mAccumulatedDigits copy] autorelease]]; +    [self clearAccumulatedInput]; +} + +- (void) clearAccumulatedInput; +{ +    [mAccumulatedDigits deleteCharactersInRange:NSMakeRange(0, [mAccumulatedDigits length])]; +     +    [self invalidateBarcodeInputTimer]; +} + +- (void) invalidateBarcodeInputTimer; +{ +    [mBarcodeInputTimer invalidate]; +    [mBarcodeInputTimer release]; +    mBarcodeInputTimer = nil; +} + +@end | 
