Restrict NSTextField input to numeric only? NSNumberformatter

Rather than subclassing anything, you can do it with an NSControlTextEditingDelegate method in your view controller. Set the delegate field of the NSTextField you want to check and then do something like the following in a controlTextDidChange: function which gets called on each character typed. Don't forget to initialize self.lastValidTimestampCount in this case. This function uses RegExLite to check the value, which may include commas.

// Make sure the user has typed a valid number so far.
- (void)controlTextDidChange:(NSNotification *)obj
{
    if (obj.object == self.timestampsCount)
    {
        NSString *countValue = timestampsCount.stringValue;
        if (! [countValue isMatchedByRegex:@"^[\\d\\,]{1,6}$"])
        {
            timestampsCount.stringValue = self.lastValidTimestampCount;
            NSBeep();
        }
        else
            self.lastValidTimestampCount = timestampsCount.stringValue;
    }
}

Here are the steps to create the same:

Just create the ANYCLASS (called SAMPLE) subclassing the NSNumberFormatter, and in the .m file write the following code...

-(BOOL)isPartialStringValid:(NSString *)partialString newEditingString:(NSString **)newString errorDescription:(NSString **) error {
    // Make sure we clear newString and error to ensure old values aren't being used
    if (newString) { *newString = nil;}
    if (error)     {*error = nil;}

    static NSCharacterSet *nonDecimalCharacters = nil;
    if (nonDecimalCharacters == nil) {
        nonDecimalCharacters = [[NSCharacterSet decimalDigitCharacterSet] invertedSet] ;
    }

    if ([partialString length] == 0) {
        return YES; // The empty string is okay (the user might just be deleting everything and starting over)
    } else if ([partialString rangeOfCharacterFromSet:nonDecimalCharacters].location != NSNotFound) {
        return NO; // Non-decimal characters aren't cool!
    }

    return YES;
}

Now in your Actual Class set the formatter to your NSTextField object like this:

NSTextField *mySampleTxtFld;

And for this set the Formatter:

SAMPLE* formatter=[[SAMPLE alloc]init]; // create SAMPLE FORMATTER OBJECT 

self.mySampleTxtFld.delegate=self;
[self.mySampleTxtFld setFormatter:formatter];

You’re done!


Here's an alternative implementation:

+ (BOOL)stringIsNumber:(NSString *)str {
   BOOL valid;
   double holder;
   NSScanner *scan = [NSScanner scannerWithString: str];
   valid = [scan scanDouble:&holder] && [scan isAtEnd];
  return valid;
}

+ (NSString *)numericStringFromString:(NSString *)string {
    NSString *digitsString = string;
    if (![YOURCLASSNAME stringIsNumber:string]) {
        NSUInteger length = [string length];
        if (length > 0) {
            digitsString = [string substringToIndex:length - 1];
            if (![YOURCLASSNAME stringIsNumber:digitsString]) {
                digitsString = [YOURCLASSNAME numericStringFromString:digitsString];
            }
        }
    }
    return digitsString;
}

Then in my controller class, I implement controlTextDidChange:

- (void)controlTextDidChange:(NSNotification *)obj {
   NSString *digitsString = [YOURCLASSNAME numericStringFromString:self.currentCellView.textField.stringValue];
   if (digitsString) {
    self.currentCellView.textField.stringValue = digitsString;
   } else {
    self.currentCellView.textField.stringValue = @"";
   }
}

The benefit of this approach is that if you paste text into your number field, this will strip out all the non numeric characters from the end of the string until you're left with just a number. Plus it supports an arbitrarily long series of digits. Some of the other approaches would not support putting in a series of digits that couldn't be held in an NSInteger.

I know this approach could certainly be improved though.


Subclass NSNumberFormatter and implement this method:

- (BOOL)isPartialStringValid:(NSString **)partialStringPtr
       proposedSelectedRange:(NSRangePointer)proposedSelRangePtr
              originalString:(NSString *)origString
       originalSelectedRange:(NSRange)origSelRange
            errorDescription:(NSString **)error