0

我有两个按钮,按玩家姓名和 ID 排序。我对每种类型都使用了两个函数。

<ul data-bind="template: { name: 'playerstmpl', foreach: players }"></ul>
<hr/>
<button data-bind="click: sortbyPlayer">Sort by Player</button>
<button data-bind="click: sortbyId">Sort by Id</button>
<script id="playerstmpl" type="text/html">
    <li> <span data-bind = "text: id"> </span>
        <input data-bind="value: FirstName" /> </li>
</script>



<script type="text/javascript">

    function Player(id, name) {
        return {
            id: ko.observable(id),
            FirstName: ko.observable(name)
            };
    }

    var viewModel = {
        players: ko.observableArray([
        new Player(64, "Yuvi"),
        new Player(22, "Gayle"),
        new Player(91, "Adam"),
        new Player(19, "Flintoff"),
        new Player(56, "Malinga")])
    };

    viewModel.sortbyPlayer = function () {
        var unsorted = viewModel.players();

        viewModel.players(unsorted.sort(viewModel.sortFunction = function (a, b) {

        return a.FirstName().toLowerCase() < b.FirstName().toLowerCase() ? -1 : 1;

        }));
    };


    viewModel.sortbyId = function () {
        var unsortedId = viewModel.players();

        viewModel.players(unsortedId.sort(viewModel.sortFunction = function (a, b) {
            return a.id() < b.id() ? -1 : 1;
        }));
    };
            ko.applyBindings(viewModel);

</script>

我试图找到在函数中传递的属性,以便我可以将两个函数合并为一个..like

viewModel.sortbyPlayer = function () {
        var unsorted = viewModel.players();

        viewModel.players(unsorted.sort(viewModel.sortFunction = function (a, b) {

            if (Player[a] == "FirstName") {            // is string
                return a.FirstName().toLowerCase() < b.FirstName().toLowerCase() ? -1 : 1;
            }
       else {
            return a.id() < b.id() ? -1 : 1;
        }

        }));
    };

无论如何要对一个功能进行排序..谢谢

4

2 回答 2

1

我会在视图模型上保留这两个属性,sortByPlayer以便sortById在您的 ViewModel 上清楚您要公开的内容。接下来,我将创建一个函数来包装排序逻辑,但允许您传入一个参数,该参数指示您想要执行哪种排序(我称之为 this sortFunction)。然后您的if语句按预期工作,如果您希望对其他属性进行排序(如果/当您的Player对象更改时),您甚至可以将其转换为switch语句:

function Player(id, name) {
    return {
        id: ko.observable(id),
        FirstName: ko.observable(name)
    };
}

var viewModel = {
    players: ko.observableArray([
            new Player(64, "Yuvi"),
            new Player(22, "Gayle"),
            new Player(91, "Adam"),
            new Player(19, "Flintoff"),
            new Player(56, "Malinga")
    ])
};

var sortFunction = function (sortBy) {
    return function () {
        var unsortedId = viewModel.players();
        viewModel.players(
            unsortedId.sort(function (a, b) {
                if (sortBy === 'player') {
                    return a.FirstName().toLowerCase() < b.FirstName().toLowerCase() ? -1 : 1;
                } else {
                    return a.id() < b.id() ? -1 : 1;
                }
            })
        );
    };
};

viewModel.sortbyPlayer = sortFunction('player');
viewModel.sortbyId = sortFunction('id');

ko.applyBindings(viewModel);
于 2013-04-24T04:58:45.073 回答
0

如果您正在寻找更通用的排序功能..您可以检查属性type..而不是它的name。这将适用于您的情况以及更一般的情况:

viewModel.sortbyPlayer = function () {
    var unsorted = viewModel.players();

    viewModel.players(unsorted.sort(viewModel.sortFunction = function (a, b) {
        if (typeof Player[a] === 'string') {            // is string
            return a.FirstName().toLowerCase() < b.FirstName().toLowerCase() ? -1 : 1;
        }
        if (typeofPlayer[a] === 'number') { 
            return a.id() < b.id() ? -1 : 1;
        }

    }));
};

您还可以使用魔法属性arguments来访问与函数一起使用的参数数组:

function someFunctionWithRandomParams(){
    if(arguments[0]){ 
        console.log('argument 1 of type'+ typeof arguments[0] +' and value'+arguments[0])
    }
    if(arguments[1]){ 
        console.log('argument 2 of type'+ typeof arguments[1] +' and value'+arguments[1])
    }
}
于 2013-04-24T05:17:49.870 回答