Knockout DateTime Picker - 默认日期不绑定

时间:2015-03-26 16:26:06

标签: datetime knockout.js knockout-mapping-plugin

我正在使用此Bootstrap DateTime Picker:http://eonasdan.github.io/bootstrap-datetimepicker/

我为DateTime创建了一个名为datepicker的自定义活页夹:

ko.bindingHandlers.datepicker = {
    init: function(element, valueAccessor, allBindingsAccessor) {
        //initialize datepicker with some optional options
        var options = allBindingsAccessor().datepickerOptions || {format: 'DD/MM/YYYY HH:mm'};
        $(element).datetimepicker(options);

        //when a user changes the date, update the view model
        ko.utils.registerEventHandler(element, "dp.change", function(event) {
            var value = valueAccessor();
            if (ko.isObservable(value)) {
                value(event.date);
            }
        });
    },
    update: function(element, valueAccessor)   {
        var widget = $(element).data("datepicker");
        //when the view model is updated, update the widget
        if (widget) {
            widget.date = ko.utils.unwrapObservable(valueAccessor());
            if (widget.date) {
                widget.setValue();
            }
        }
    }
};

我的模型是JSON,因为我将C#dateTime转换为JSON我得到了这个日期:"/Date(1427368178393)/"

            <div class="col-md-4">
                <div class="form-group">
                    <label class="control-label">Quote Date</label>
                    <div class='input-group date dateTimes'>
                        <input type="text" class="form-control" data-bind="datepicker: QuoteDateTime" />
                        <span class="input-group-addon"><span class="glyphicon glyphicon-calendar"></span></span>
                    </div>
                </div>
            </div>

但问题是,第一次页面加载,即使QuoteDateTime有一个值,它没有显示在DateTimePicker中。

这是JSFiddle:

https://jsfiddle.net/mdawood1991/ezv7qxbt/

我需要在DateTime Picker中显示第一个值。

修改

问题是我的ViewModel有一个住宿列表

这是我的viewModel的控制台日志

enter image description here

我可以将每个dateTime转换为你提出的时刻,但是我不应该使用knockout Mapping Plugin:http://knockoutjs.com/documentation/plugins-mapping.html

这是循环浏览viewModel然后将alld Datetimes转换为moment对象的唯一解决方案

3 个答案:

答案 0 :(得分:5)

帮助之后来自@David和@Bryant我改变了我的代码,用时刻对象日期更新了viewModel。

问题是日期只是从日期时间选择器中选择日期后才正确回发。所以我更改了代码,因此在Initialized时我更新了viewModel。它现在正在运作。

这是DateTime Picker

http://eonasdan.github.io/bootstrap-datetimepicker/

我使用了以下datepicker绑定。

JSFiddle Working:http://jsfiddle.net/mdawood1991/cL9k2sbe/

我的解决方案:

ko.bindingHandlers.datepicker = {
    init: function (element, valueAccessor, allBindingsAccessor, viewModel, bindingContext) {
        //initialize datepicker with some optional options
        var options = {
            format: 'DD/MM/YYYY HH:mm',
            defaultDate: valueAccessor()()
        };

        if (allBindingsAccessor() !== undefined) {
            if (allBindingsAccessor().datepickerOptions !== undefined) {
                options.format = allBindingsAccessor().datepickerOptions.format !== undefined ? allBindingsAccessor().datepickerOptions.format : options.format;
            }
        }

        $(element).datetimepicker(options);

        //when a user changes the date, update the view model
        ko.utils.registerEventHandler(element, "dp.change", function (event) {
            var value = valueAccessor();
            if (ko.isObservable(value)) {
                value(event.date);
            }
        });

        var defaultVal = $(element).val();
        var value = valueAccessor();
        value(moment(defaultVal, options.format));
    },
    update: function (element, valueAccessor, allBindingsAccessor, viewModel, bindingContext) {
        var thisFormat = 'DD/MM/YYYY HH:mm';

        if (allBindingsAccessor() !== undefined) {
            if (allBindingsAccessor().datepickerOptions !== undefined) {
                thisFormat = allBindingsAccessor().datepickerOptions.format !== undefined ? allBindingsAccessor().datepickerOptions.format : thisFormat;
            }
        }

        var value = valueAccessor();
        var unwrapped = ko.utils.unwrapObservable(value());

        if (unwrapped === undefined || unwrapped === null) {
            element.value = new moment(new Date());
            console.log("undefined");
        } else {
            element.value = unwrapped.format(thisFormat);
        }
    }
};

答案 1 :(得分:3)

您的更新功能根本不起作用。使其工作将解决您的问题并导致observable的更新反映在控件中。

  1. 使用$(element).data("DateTimePicker")
  2. 将日期字符串转换为moment.js对象
  3. 使用date()函数传递它。
  4. 代码:

    update: function(element, valueAccessor)   {
        var widget = $(element).data("DateTimePicker");
    
        if (widget) {
            var date = ko.utils.unwrapObservable(valueAccessor());
    
            if (typeof date === "string") {
                // convert c# string to momentjs object
                date = moment(date);
            }
    
            widget.date(date);
        }
    }
    

    编辑 - 选项2:将所有内容保留为moment.js对象

    然而,您的代码的一个问题是字符串被传递到observable,但是每当这个自定义绑定更新observable时它就会成为一个moment.js对象(input -> stringoutput -> Moment)。要解决此问题,我建议用户将moment.js对象传递给observable(input -> Momentoutput -> Moment)。

    首先,在填充视图模型时将c#字符串转换为moment.js对象:

    self.QuoteDateTime = ko.observable(moment(data.QuoteDateTime));
    

    其次,更改您的更新函数以处理moment.js对象而不是字符串:

    update: function(element, valueAccessor)   {
        var widget = $(element).data("DateTimePicker");
        //when the view model is updated, update the widget
        if (widget) {
            var date = ko.utils.unwrapObservable(valueAccessor());
            widget.date(date);
        }
    }
    

    你可以在这个jsfiddle中看到这个工作正常:

    https://jsfiddle.net/ezv7qxbt/21/

答案 2 :(得分:1)

我认为您需要使用default date option.

var options = allBindingsAccessor().datepickerOptions || {
                format: 'DD/MM/YYYY HH:mm',
                defaultDate: valueAccessor()()
            };

更新了JSFiddle: https://jsfiddle.net/3jqL8s1m/