0

我想编写一个通用方法来验证所有 UIControls,如 NSTextField、NSTextView 等。如果任何必填字段为空,则应显示一个合并的警报,其中第一个控件作为焦点/第一响应者。

我已经实现了这样的东西:

-(NSInteger)lengthAfterTrimmingSpaces:(NSString *)string{
    return [[string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]] length];
}


-(NSDictionary *)createWarningMessageForMandatoryFields:(NSArray *)uiObjects{

    NSMutableString *warningString=[NSMutableString stringWithString:@"Enter "];
    id firstInvalidatedControl=nil;

    for (NSDictionary *object in uiObjects) {
        NSString *key=[object allKeys][0];
        id control=object[key];
        if ([control isKindOfClass:[NSTextField class]]) {
            if ([self lengthAfterTrimmingSpaces:[control stringValue]]==0){
                [warningString appendString:key];
                [warningString appendString:@", "];

                if (firstInvalidatedControl==nil) {
                    firstInvalidatedControl=control;
                }
            }
        }
        else if ([control isKindOfClass:[NSTextView class]]) {
            if ([self lengthAfterTrimmingSpaces:[control string]]==0){
                [warningString appendString:key];
                [warningString appendString:@", "];
                if (firstInvalidatedControl==nil) {
                    firstInvalidatedControl=control;
                }
            }
        }
    }

    if (firstInvalidatedControl==nil) {
        return @{@"warningString":@"Success"};
    }
    else{

        warningString =[[warningString substringToIndex:[warningString length] - 2]mutableCopy];
        return @{@"warningString":warningString, @"control":firstInvalidatedControl};
    }
}


-(BOOL)validateMandatoryFields{
    NSMutableArray *uiObjects=[NSMutableArray array];

    [uiObjects addObject:@{@"Segment Name":self.segmentName}];
    [uiObjects addObject:@{@"Code":self.code}];
    [uiObjects addObject:@{@"Desciption":self.description}];

    NSDictionary *warningAndControl=[self createWarningMessageForMandatoryFields:uiObjects];
    if ([warningAndControl[@"warningString"] isEqualToString:@"Success"]) {
        return YES;
    }
    else{
        [[self window] makeFirstResponder:warningAndControl[@"control"]];
        NSRunAlertPanel(@"Warning", warningAndControl[@"warningString"], @"OK", nil, nil);
        return NO;
    }
}

- (IBAction)save:(id)sender {
    NSLog(@"%d",[self validateMandatoryFields]);
}

现在我想让它更加通用,因为对于每种我都在进行类内省,然后检索值(stringValue/string 等)。

对此的任何其他建议也表示赞赏。

4

2 回答 2

0

您基于将呈现给用户的字符串内容的逻辑(确定是否存在错误)。你不应该。相反,将逻辑基于键 ( control),因为它们是不可变的。与此一起,您应该使用NSLocalizedString将来验证代码以供重用。

对于自省,我会考虑将这些方法放入一个实用程序类中,并将一些类别也添加到该类中。这些类别将为每个子类添加一个方法,NSControl目的是允许您stringValue通过始终调用具有相同名称的方法来获取它们。因为NSTextField你不需要一个类别。对于NSTextView类别将是:

@implementation NSTextView (MyStringValue)

- (NSString *)stringValue
{
    return [self string];
}

@end

您的代码仍应验证该类是否响应stringValue选择器,如果没有,则无法验证。

于 2013-05-11T07:40:42.013 回答
0

将控件绑定到查看控制器的属性可能会有所帮助。

@interface MyViewController

@property(nonatomic,copy) NSString *string;
@property(nonatomic,copy) NSDate *date;
@property(nonatomic,copy) NSNumber *number;

@end

现在您将控件值绑定到文件所有者的string//属性datenumber

@implementation MyViewController

- (void)testForm
{
    for (NSString *key in @[@"string", @"date", @"number"]) {
        id value = [self valueForKey:key];
        if (value == nil || [value isEqual:@""] || [value isEqual:@(0.0)] || ...) {
            // empty field
        }
    }
}

请注意,您在此处使用真实数据,即如果您有日期/数字字段,则不会检查其 stringValue(即格式化程序的工作)的完整性,而是检查已转换/格式化的值。

于 2013-05-11T11:42:44.573 回答