3

我正在学习淘汰赛、Durandal 和 Breeze,感谢John Papa 的 JumpStart SPA 视频,这是一种乐趣。所以我检查了这个项目的代码,现在我正在尝试更改验证机制。

此时,在保存更改时,如果保存失败,我们会收集错误并显示带有错误恢复的 toast。

在此处输入图像描述

我想要实现的是:

  • 能够在视图中以红色(背景色)标记无效字段

  • 当显示带有验证错误恢复的吐司时(例如:'保存失败:需要事务Nr')我想用更友好的名称替换属性名称(例如'保存失败:需要事务编号')

这是 datacontext.js 中验证的部分代码:

var saveChanges = function () {
    return manager.saveChanges()
        .then(saveSucceeded)
        .fail(saveFailed);

    function saveSucceeded(saveResult) {
        log('Saved data successfully', saveResult, true);
    }

    function saveFailed(error) {
        var msg = 'Save failed: ' + getErrorMessages(error);
        logError(msg, error);
        error.message = msg;
        throw error;
    }
};

function getErrorMessages(error) {
    var msg = error.message;
    if (msg.match(/validation error/i)) {
        return getValidationMessages(error);
    }
    return msg;
}

function getValidationMessages(error) {
    try {
        //foreach entity with a validation error
        return error.entitiesWithErrors.map(function (entity) {
            // get each validation error
            return entity.entityAspect.getValidationErrors().map(function (valError) {
                // return the error message from the validation
                return valError.errorMessage;
            }).join('; <br/>');
        }).join('; <br/>');
    }
    catch (e) { }
    return 'validation error';
}

有人能指出我正确的方向吗?

提前致谢。


编辑:

要重现该问题:单击左侧边栏上的 Transports + recherche avancee + 列表中的任何项目 + 在右侧:清除一些输入(如 Numero d'affaire 如下截图所示),然后单击“Enregistrer”。然后调用保存按钮。在那里我需要检查 ko.validation.group 是否有无效输入,但它不起作用。

在此处输入图像描述

4

1 回答 1

8

前段时间,我发布了一个辅助函数,它扩展了 observable 属性以添加基于微风验证器的验证。使用该助手,您可以在无效输入上实现红色:

翻译微风验证消息

第二个问题我无法帮助您,我知道您可以自定义验证消息,但我认为使用默认验证器您无法设置友好名称以显示在消息上。

更新:

我要做的第一件事是创建一个公开函数的辅助模块(更新版本):

define([],
function () {
"use strict";
var foreignKeyInvalidValue = 0;

function addValidationRules(entity) {
    var entityType = entity.entityType,
        i,
        property,
        propertyName,
        propertyObject,
        validators,
        u,
        validator,
        nValidator;

    if (entityType) {
        for (i = 0; i < entityType.dataProperties.length; i += 1) {
            property = entityType.dataProperties[i];
            propertyName = property.name;
            propertyObject = entity[propertyName];
            validators = [];

            for (u = 0; u < property.validators.length; u += 1) {
                validator = property.validators[u];
                nValidator = {
                    propertyName: propertyName,
                    validator: function (val) {
                        var error = this.innerValidator.validate(val, { displayName: this.propertyName });
                        this.message = error ? error.errorMessage : "";
                        return error === null;
                    },
                    message: "",
                    innerValidator: validator
                };
                validators.push(nValidator);
            }
            propertyObject.extend({
                validation: validators
            });
        }

        for (i = 0; i < entityType.foreignKeyProperties.length; i += 1) {
            property = entityType.foreignKeyProperties[i];
            propertyName = property.name;
            propertyObject = entity[propertyName];

            validators = [];
            for (u = 0; u < property.validators.length; u += 1) {
                validator = property.validators[u];
                nValidator = {
                    propertyName: propertyName,
                    validator: function (val) {
                        var error = this.innerValidator.validate(val, { displayName: this.propertyName });
                        this.message = error ? error.errorMessage : "";
                        return error === null;
                    },
                    message: "",
                    innerValidator: validator
                };
                validators.push(nValidator);
            }
            propertyObject.extend({
                validation: validators
            });
            if (!property.isNullable) {
                //Bussiness Rule: 0 is not allowed for required foreign keys
                propertyObject.extend({ notEqual: foreignKeyInvalidValue });
            }
        }
    }
}

return {
    addValidationRules: addValidationRules
};
});

然后,我为每个微风实体类型(http://www.breezejs.com/documentation/extending-entities)定义了一个初始化程序。例子:

define(['app/validatorHelper', 'knockout'],
function (vHelper, ko) {
"use strict";
var constructor = function () {
},

    initializer = function indicadorInitializer(entity) {
        vHelper.addValidationRules(entity);
    };

return {
    constructor: constructor,
    initializer: initializer
};
});

最后,在某个地方(我在我的数据服务模块内的 init 函数上执行此操作),我正在注册初始化程序(http://www.breezejs.com/documentation/extending-entities):

//store comes from: manager = breezeconfig.createManager(),
//    store = manager.metadataStore,
store.registerEntityTypeCtor("Palanca", domain.palanca.constructor, domain.palanca.initializer);

在获取元数据之前,我正在完成所有这些工作。

我希望这会对你有所帮助。

更新 2:

我发现了问题,你的淘汰验证版本不是最后一个。

在从http://ericmbarnard.github.com/Knockout-Validation/下载的文件中,第 349 行是:

exports.rules[ruleName] = ruleObj;

在您的文件中,等效行(函数 addAnonymousRule )是:

ko.validation.rules[ruleName] = {
    validator: ruleObj.validator,
    message: ruleObj.message || 'Error'
};

我认为最新版本应该可以工作。

更新 4:

这是要保存的代码:

vmAddPalanca.prototype.saveChangesCmd = ko.asyncCommand({
    execute: function (palanca, complete) {
        var validationErrors = ko.validation.group(palanca);
        if (validationErrors().length === 0) {
            dataservice.saveChanges()
            .then(saveChangesSuccess)
            .fail(saveChangesFail)
            .fin(complete);
        } else {
            validationErrors.showAllMessages(true);
            toastr.error("Debe corregir los errores antes de poder guardar");
            complete();
        }
    },
    canExecute: function (isExecuting) {
        return !isExecuting && dataservice.hasChanges();
    }
});
于 2013-03-26T14:39:58.917 回答