使用Vue.js的bootstrap-datetimepicker无法正常工作

7

我正在尝试将bootstrap-datepicker与Vue.js一起使用,尝试创建一个简单的日期选择器组件。问题是,当我写入输入框时,组件的数据会更新。但是当我使用日期选择器GUI时,它不起作用。数据没有更新。为什么?到底缺少了什么?

这是我的代码:

Vue.component('picker', {
  'props': {
    'date': {
      'twoWay': true
    }
  },
  'template': '\
      <div class="form-group">\
          <div class="input-group date datepicker">\
              <input type="text" class="form-control" v-bind:value="this.date">\
              <span class="input-group-addon">\
                  <span class="glyphicon glyphicon-calendar"></span>\
              </span>\
          </div>\
      </div>',
  'watch': {
    'date': function(value) {
      console.log('Updated!');
    }
  },
  'mounted': function() {
    $(this.$el.firstChild).datetimepicker();
  }
});

app = new Vue({
  'el': '#app',
  'data': {}
});
<!DOCTYPE html>
<html>

<head>
  <title></title>
  <!-- JQuery -->
  <script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>

  <!-- Transition ve Collapse -->
  <script src="collapse.js"></script>
  <script src="transition.js"></script>

  <!-- Moment JS -->
  <script src="moment-with-locales.min.js"></script>

  <!-- Latest compiled and minified CSS -->
  <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">

  <!-- Optional theme -->
  <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap-theme.min.css" integrity="sha384-rHyoN1iRsVXV4nD0JutlnGaslCJuC7uwjduW9SVrLvRYooPp2bWYgmgJQIXwl/Sp" crossorigin="anonymous">

  <!-- Latest compiled and minified JavaScript -->
  <script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.min.js" integrity="sha384-Tc5IQib027qvyjSMfHjOMaLkfuWVxZxUPnCJA7l2mCWNIpG9mGCD8wGNIcPD7Txa" crossorigin="anonymous"></script>

  <!-- Datetime picker -->
  <script src="https://raw.githubusercontent.com/Eonasdan/bootstrap-datetimepicker/master/build/js/bootstrap-datetimepicker.min.js"></script>
  <link rel="stylesheet" type="text/css" href="https://raw.githubusercontent.com/Eonasdan/bootstrap-datetimepicker/master/build/css/bootstrap-datetimepicker.min.css">
  <script src="https://unpkg.com/vue/dist/vue.js"></script>
</head>

<body>
  <div class="container">
    <div id="app" class="row">
      <div class='col-sm-6'>
        <picker></picker>
      </div>
    </div>
  </div>
  <script type="text/javascript">
  </script>
</body>
<script src="app.js"></script>

</html>


我看到选择器组件接受 date 属性,但在模板的 <picker></picker> 组件中找不到它的定义。 - Belmin Bedak
我看不到你在数据中定义了日期。 - Saurabh
@BelminBedak 我在HTML部分没有给date prop一个初始值。 @Saurabh 它是一个prop。所以我不需要在Vue实例中定义它。对吗? 更新的代码。 - nejdetckenobi
当运行您的代码片段时,我收到了$(...).datetimepicker不是一个函数的错误信息。 - Roy J
@RoyJ:是的,我也遇到了那个错误。我不知道为什么会这样响应,但当我运行它时,它运行得很好。无论如何,感谢您的努力。 - nejdetckenobi
6个回答

5

我没有找到比失去焦点(blur)更好的方法。

这是我的代码:

Vue.component('picker', {
    'props': ['date'],
    'template': '\
        <div class="form-group">\
            <div class="input-group date datepicker">\
                <input type="text" class="form-control" :value="this.date" v-on:focus="datepick_dialog_open" v-on:blur="select_by_lose_focus">\
                <span class="input-group-addon">\
                    <span class="glyphicon glyphicon-calendar"></span>\
                </span>\
            </div>\
        </div>',
    'mounted': function() {
        $(this.$el.firstChild).datetimepicker();
    },
    'methods': {
        'datepick_dialog_open': function() {
            this.$el.firstChild.children[1].click();
        },
        'select_by_lose_focus': function() {
            console.log('AHOY!');
            this.date = this.$el.firstChild.firstChild.value;
        }
    }
});

app = new Vue({
    'el': '#app',
    'data': {}
});

重点是,如果您的元素没有失去焦点,您真的无法使用它。它会在输入框失去焦点时更新数据。我不知道有更好的方法。 v-on:focus 是无关紧要的。它只是在输入框获得焦点时打开日期时间选择器UI。
主要工作由 select_by_lose_focus 完成。

2
另一个使用 $emit 和组件上的 v-model 的例子。此页面描述了为什么需要 value 和 $emit 来支持 v-model。我需要一个表单元素名称和可选的必填类以进行验证。 https://v2.vuejs.org/v2/guide/components.html#Form-Input-Components-using-Custom-Events
  <picker name="start_date_1_1" required="1" v-model="res.start_date"></picker>
  
  Vue.component('picker', {
    props: ['name', 'value', 'required'],
    template: '\
      <div class="form-group">\
          <div class="input-group date">\
              <input :name="name" type="text" class="form-control" v-bind:class="{required: required}"  v-bind:value="value">\
              <span class="input-group-addon">\
                  <span class="glyphicon glyphicon-calendar"></span>\
              </span>\
          </div>\
      </div>',
    mounted: function() {
      var self = this;
      var picker = $(this.$el.firstChild).datepicker({autoclose: true});
      picker.on('changeDate', function(e) {
        self.$emit('input', e.date.toLocaleDateString("en-US", {day: '2-digit', month: '2-digit', year: 'numeric'}));
      });
    }
  });

mounted() { var self = this; var picker = $(".dtp").timepicker(); picker.on('changeTime.timepicker', function(e) { self.form[e.target.name] = e.time.value; }); }, - Pasan Bhanu Guruge

2
在Vue对象初始化之前添加:


    Vue.directive('vueinput', {
        twoWay: true,
        bind: function (el, binding, vnode) {
            $(el).on("change", (e) => {
                el.dispatchEvent(new Event('input', { target: e.target }));
            });
        },
    });

然后将v-vueinput="[uniqueString]"指令添加到您的日期时间输入框中:

<div data-start-view="4" data-min-view="2" class="input-group date datetimepicker">
    <input type="text"
           v-model="payment.pay_date"
           v-vueinput="payment.pay_date">
    <div class="input-group-append">
        <button class="btn btn-primary"><i class="icon-th mdi mdi-calendar"></i></button>
    </div>
</div>

我建议将与v-model相同的值(例如我的示例中的'payment.pay_date')放入此指令中。

您可以在页面上使用无限数量的输入实体来使用这些指令。


0

通过在日期更改时调度输入事件,使其正常工作。

$('.with-calendar').datetimepicker({
    weekStart: 1,
    language: 'de',
    bootcssVer: 3,
    format: "yyyy-mm-dd hh:ii:00",
    viewformat: "yyyy-mm-dd hh:ii:00",
    autoclose: true
}).on('changeDate', function (event) { // Communicate datetimepicker result to vue
    let inputFields = event.target.getElementsByTagName('input'); // depends on your html structure
    for (let i = 0; i < inputFields.length; i++) {
        inputFields[i].dispatchEvent(new Event('input', {'bubbles': true}));
    }
});
<template>
...
<div class="form-group">
    <label class="col-sm-3 control-label">Date and Time</label>
    <div class="col-sm-6">
        <div class="input-group date with-calendar">
            <input type="text" class="form-control" v-model="start_datetime"/>
            <span class="input-group-addon">
                <span class="glyphicon glyphicon-calendar"></span>
            </span>
        </div>
    </div>
</div>
...
</template>


0

我遇到了这个问题,通过在初始化Vue之后初始化日期选择器来解决它。例如:

var vm = new Vue({
    el: '#myForm',
    data: {
        field1: "",
        field2: ""
    }
});

$(".date-picker").datepicker({
    todayHighlight: true,
    autoclose: true,
    closeOnDateSelect: true
});

这不是一个好的初始化方式。如果你没有正确地初始化它,就不能将其用作对象。然而,你放在初始化器中的参数很有趣。 - nejdetckenobi

0

感谢@nejdetckenobi的回答,如果只选择日期,则模糊方法有效。但是,如果选择日期和时间,我发现模糊方法效果不佳。

相反,我遵循了Vue.js包装组件的示例。它利用on('change', function(){})$emit

下面的代码在我的设置中对我起作用,其中关键部分位于您的vue组件声明的mounted部分。

Vue.component('datetime', {
  props: ['value'],
  template: '<input class="form-control" id="override-datetime-field"></input>',
  mounted: function () {
    var vm = this
    $(this.$el)
      .datetimepicker()
      .val(this.value === '' ? '' : moment(this.value).format('DD-MMM-YYYY, h:mm a'))
      .trigger('change')
      .on('change', function () {
        vm.$emit('input', this.value)
      })
  },
  watch: {
    value: function (value) {
      $(this.$el).val(value).trigger('change');
    }
  }
} 

关键部分在这里:

$(an-element).datetimepicker().trigger('change')
.on('change', function () {
  vm.$emit('input', this.value)
})

原始代码示例:https://v2.vuejs.org/v2/examples/select2.html


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接