0

我无法使用 knockout.js 设置自定义验证规则以检查用户名是否已存在。据我了解,如果返回为真,则没有错误,否则设置错误。

是自定义验证的示例

//val is the username in question and searchType is the type of search(username or email)
function checkValue(val, searchType){
  if(searchType == 'userName'){
    $.post("/users/check_if_exists",{ 'type':'username', 'value': val },function(data) {
        var info = JSON.parse(data);
        if(info.username_availability == "available"){
            return searchType;
                            //I know this is working because I've alerted the searchtype here and it displays properly
        }
        else{
            return "unavailable";
        }
    });
  }
} 

ko.validation.rules['checkIfExists'] = {
    validator: function (val, searchType) {
        return searchType == checkValue(val, searchType); //if the username is availble, the searchType is returned back so it would return searchType == searchType which should be true meaning there are no errors
    },
    message: 'This username is taken, please select another.'
};
ko.validation.registerExtenders();

我检查了网络选项卡,POST 正在返回正确的值。如果该值可用,则返回 searchType。这样,它比较 searchType == searchType 应该是真的。然而,事实并非如此。

还有其他方法可以完成我想做的事情吗?

更新

这是我现在所拥有的

function checkValue(val, searchType, callback) {
     var callback = function(data) {
        return true;
    }
    $.post("/users/check_if_exists", { 'type':'username', 'value': val }, function(data) {
        info = JSON.parse(data);
        if(info.username_availability == "available"){
            callback(true);
        } else {
            callback(false);
        }
    }); 
}

ko.validation.rules['checkIfExists'] = {
    async: true,
    validator: function (val, searchType) {
        alert(checkValue(val, searchType));//returns undefined
        return checkValue(val, searchType);
    },
    message: 'This username is taken, please select another.'
};
ko.validation.registerExtenders();
4

3 回答 3

2

ko.validation 调用您的验证函数。关于将成功/失败状态传递回 ko.validation 的方式,有两种类型的验证函数:直接返回 true/false 方式,或“异步”方式。

异步方式之所以存在,只是因为 $.ajax 存在,它基本上就是这样:而不是返回一个值(这是不可能的,因为你正在使用 $.ajax 调用)你必须在 ajax 响应返回后以某种方式通知 ko.validation到浏览器,对吧?

所以编写这个库的聪明人用一个额外的参数调用你的验证函数,一个函数(回调),当响应可用时你必须调用它。

function checkValue(val, searchType, callback){
  if(searchType == 'userName'){
    $.post("/users/check_if_exists",{ 'type':'username', 'value': val },function(data) {
        var info = JSON.parse(data);
        if(info.username_availability == "available"){
            callback(true);
                            //I know this is working because I've alerted the searchtype here and it displays properly
        }
        else{
            callback(false);
        }
    });
  }
} 


ko.validation.rules['checkIfExists'] = {
    async: true,
    validator: checkValue,
    message: 'This username is taken, please select another.'
};
ko.validation.registerExtenders();
于 2013-08-22T22:12:01.923 回答
1

让我把你的代码变成更易读的东西:

function checkValue(val) {
    var callback = function(data) {
        return 'bar';
    }

    $.post('url', 'data', callback); 
}

var x = checkValue('foo');

如果不是这种情况,您期望 x 设置为 'bar'。我用“回调”替换了你的匿名函数,所以你更好地理解从它返回的东西并不意味着“checkValue”会返回任何东西。之后,您的下一个问题是 AJAX 调用是异步的。

验证器函数接受三个参数:值、额外参数和回调函数。回调函数参数旨在帮助您解决这种情况。

在 $.post 的成功回调中,您只需为验证器函数调用该回调函数,并将 true/false 作为参数传递。

function checkValue(val, params, callback) {
    $.post('url', 'data', function(data) {
        if(data === 'bar') {
            callback(true);
        } else {
            callback(false);
        }
    }); 
}
于 2013-08-22T21:28:25.230 回答
0

正如您所写的,checkValue 总是返回未定义的。从回调函数内部调用“return”只会设置该回调函数(而不是“外部函数”)的返回值。如果验证器函数需要立即返回值(就像淘汰赛似乎所做的那样),您将无法异步获取该数据。

于 2013-08-22T21:27:09.533 回答