设置一个knockout.js observable 而不触发订阅

Tug*_*ain 3 javascript knockout.js

如何在不触发任何订阅的情况下设置 observable 属性?

我有一个场景是页面加载,一个ajax调用来获取一些数据,数据被循环,然后当前选择的项目被设置为一个可观察的。我希望能够在没有任何订阅的情况下设置这个 observable,因为第一次设置这个 observable 被认为是它的初始状态,并且订阅不应该在初始状态下执行。

function PlanViewModel() {
    var self = this;

    self.plans = ko.observableArray();
    self.selectedItem = ko.observable();

    self.getAllPlans = function () {
        $.ajax({
            url: "/Backoffice/Home/GetAllPlans",
            type: "POST",
            data: {},
            context: this,
            success: function (result) {
                var planList = this.plans;
                // clear the plan list
                planList.removeAll();
                $.each(result.plans, function () {
                    var planDetail = new PlanDetail(this, self);
                    if (this.IsSelected) {
                        self.selectedItem(planDetail); // how do I set this without the subscriptions firing?
                    }
                    planList.push(planDetail);
                });
            },
            error: function (result) {
                alert("An error occured getting plans.");
            }
        });
    }

    self.selectedItem.subscribe(function (newItem) {
        newItem.repositoryUpdateSelectedPlan();
    } .bind(self));
}
Run Code Online (Sandbox Code Playgroud)

mil*_*ose 5

你可以像这样重构你的代码:

function PlanViewModel() {
    var self = this;

    self.plans = ko.observableArray();

    self.getAllPlans = function () {
        $.ajax({
            // …
            success: function (result) {
                // …
                $.each(result.plans, function () {
                    var planDetail = new PlanDetail(this, self);
                    if (this.IsSelected) {
                        self.selectedItem = ko.observable(planDetail);
                    }
                    planList.push(planDetail);
                });
                if (self.selectedItem === undefined) { 
                    self.selectedItem = ko.observable(); 
                }
                self.selectedItem.subscribe(function (newItem) {
                    newItem.repositoryUpdateSelectedPlan();
                }.bind(self));
            },
            // …
        });
    }
}
Run Code Online (Sandbox Code Playgroud)

也就是说,只有达到您想要的初始状态后才开始淘汰赛 。