【问题标题】:Knockout DateTime Picker - Default Date not Binding淘汰日期时间选择器 - 默认日期不绑定
【发布时间】:2015-05-30 18:51:04
【问题描述】:

我正在使用这个 Bootstrap 日期时间选择器:http://eonasdan.github.io/bootstrap-datetimepicker/

我为 DateTime 创建了一个名为 datepicker 的自定义 Binder:

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/

我需要在日期时间选择器中显示第一个值。

编辑

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

这是我的 viewModel 的控制台日志

按照您的建议,我可以将每个 dateTime 转换为时刻,但我不应该使用淘汰赛映射插件:http://knockoutjs.com/documentation/plugins-mapping.html

这是循环通过我的视图模型然后将所有日期时间转换为时刻对象的唯一解决方案

【问题讨论】:

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


    【解决方案1】:

    在 @David 和 @Bryant 的帮助下,我更改了代码以使用时刻对象日期更新 viewModel。

    问题是日期只有在从日期时间选择器中选择日期后才能正确回发。所以我更改了代码,所以在初始化时我更新了 viewModel。它现在正在工作。

    这是日期时间选择器

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

    我使用了以下日期选择器绑定。

    JSFiddle 工作: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);
            }
        }
    };
    

    【讨论】:

      【解决方案2】:

      您的更新功能根本不起作用。让它工作将解决您的问题并导致对可观察对象的更新反映在控件中。

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

      代码:

      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 -&gt; stringoutput -&gt; Moment)。为了解决这个问题,我建议让用户将 moment.js 对象传递给 observable(input -&gt; Momentoutput -&gt; 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/

      【讨论】:

      • 顺便说一句,您还需要更改您的更改事件处理程序以将可观察到的淘汰赛设置为格式为/Date(1427368178393)/ 的字符串(这样它是一致的,但我个人会将其更改为输入和输出一个 Date 对象)
      • @DawoodAwan 查看我的更新和工作示例(我意识到你这里有 moment.js,所以最好使用它)。
      • @BryantMiano 哈哈,没关系。
      • 大卫的方法比我的回答干净得多。使用 moment.js 是要走的路。
      • @DavidSherret 问题是我有一个对它们有 DateTime 的住宿数组,然后每次发送 Ajax 请求时我都必须遍历我的 viewModel。在问题中查看我的编辑
      【解决方案3】:

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

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

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

      【讨论】:

      • 干杯。这就是我所缺少的
      • 这不会将正确的日期时间发布到服务器。它正在发布 01/01/0001 如果您不选择日期
      猜你喜欢
      • 2015-09-22
      • 1970-01-01
      • 1970-01-01
      • 2021-03-14
      • 2014-01-02
      • 1970-01-01
      • 2011-10-02
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多