2

我有一个通过 AJAX 获取 JSON 并创建新任务的视图模型,但 Knockout 不断给我一个绑定错误。

如果我直接在我的视图模型中对来自服务器的数据进行硬编码,我不会收到任何错误。

我的视图模型创建了一个新任务,它有一个 id、一个问题和一些备选方案,它本身有一个文本和正确的布尔标志。

以下代码可以正常工作:

function Task(data) {
    var self = this;

    self.id = data.id;
    self.question = ko.observable(data.question);

    var alts = new Array();
    $(data.alternatives).each(function(index){
        alts.push(new Alternative(data.alternatives[index].alternative, data.alternatives[index].correct));
    });

    self.alternatives = ko.observableArray(alts);
}
function Alternative(alternativeText, correctAnswer) {
    var self         = this;
    self.alternative = ko.observable(alternativeText);
    self.correct     = ko.observable(correctAnswer);
}
function TaskViewModel() {
    var self = this;

    var data = {
        id: 5,
        question: 'test',
        alternatives: [{
            alternative: 'alt 1',
            correct: false
        },{
            alternative: 'alt 2',
            correct: true
        },{
            alternative: 'alt 3',
            correct: false
        }]
    };

    self.task = new Task(data);
}

但是,如果我用来自服务器的真实数据交换硬编码data变量:

function TaskViewModel() {
    var self = this;

    $.getJSON('/data', function(data){
        self.task = new Task(data);
    });
}

淘汰赛给了我这个错误:

Error: Unable to parse bindings.
Message: ReferenceError: Can't find variable: task;
Bindings value: value: task.question

URL 中的数据如下所示:

{"id":5,"question":"test","alternatives":[{"alternative":"alt 1","correct":false},{"alternative":"alt 2","correct":true},{"alternative":"alt 3","correct":false}]}

我似乎无法弄清楚为什么这不起作用:/

4

2 回答 2

6

应用绑定时,您的视图模型实际上没有task属性。你需要给它一些绑定的东西。

有几种方法可以处理这个问题。

可能最简单的方法是使taskobservable 并将其设置为 ajax 调用的结果。您可能需要调整绑定以应对此更改。

function TaskViewModel() {
    var self = this;
    self.task = ko.observable();

    $.getJSON('/data', function(data){
        self.task(new Task(data));
    });
}

更灵活的选择是为您的Task对象添加单独的初始化方法并设置任务(未初始化)。然后作为ajax调用的结果,调用初始化方法对其进行初始化。当然,您必须调整任务对象的初始化代码。

function TaskViewModel() {
    var self = this;
    self.task = new Task();

    $.getJSON('/data', function(data){
        self.task.init(data);
    });
}

function Task() {
    var self = this;
    self.id = ko.observable();
    self.question = ko.observable();
    self.alternatives = ko.observableArray();

    self.init = function (data) {
        self.id(data.id);
        self.question(data.question);
        self.alternatives(ko.utils.arrayForEach(data.alternatives, function (item) {
            return new Alternative(item.alternative, item.correct);
        }));
    };
}
于 2012-11-11T18:23:40.860 回答
4

这篇文章包含了在 source 为 null 时可以处理绑定的多种方法。

源为空/未定义时的 KnockoutJS 绑定

如果您对不可绑定的 UI 消失感到满意,我建议您使用with

如果selectedItem为 null 则该元素甚至不会显示。

<div data-bind="with: selecteditem">
    <form>
        <fieldset>
            <div>
                <label>first name</label>
                <input data-bind="value: firstname"></input>
            </div>
            <div>
                <label>lasst name</label>
                <input data-bind="value: lastname"></input>
            </div>
        </fieldset>
        <div>
            <a href="#" data-bind="click: $root.savechanges">Save</a>
        </div>
    </form>
</div>
于 2013-04-26T20:49:43.710 回答