我正在尝试将bootstrap-datepicker与Vue.js一起使用 试图创建一个简单的datepicker组件。 问题是,当我写入输入时,组件的数据会更新。 但是当我使用datepicker 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>
答案 0 :(得分:5)
我找不到比失去焦点(模糊)更好的方法。
这是我的代码:
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': {}
});
重点是,如果你的元素没有失去焦点,你真的不能使用它。当输入失去焦点时,它会更新数据。 IDK更好的方式。
v-on:focus
无关紧要。它只是在输入获得焦点时打开日期时间选择器UI。
主要工作由select_by_lose_focus
完成。
答案 1 :(得分:1)
在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')
您可以在一页上使用此指令使用无限量的输入实体。
答案 2 :(得分:0)
感谢@nejdetckenobi的答案,如果仅选择日期,模糊方法可行。但是,如果选择两者日期&amp;时间,我发现模糊方法效果不好。
相反,我遵循了Vue.js包装器组件示例。它使用on('change', function(){})
和$emit
。
以下代码(适用于我的设置)中包含日期格式,关键部分位于mounted
声明的vue component
部分。
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)
})
答案 3 :(得分:0)
另一个在组件上使用$ emit和v-model的示例。此页面描述了为支持v-model需要value和$ emit的原因。我需要一个表单元素名称和可选的必需类来进行验证。
https://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'}));
});
}
});
答案 4 :(得分: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}));
}
});
&#13;
<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>
&#13;
答案 5 :(得分:0)
解决了这个问题,并在初始化Vue之后通过初始化日期选择器使其工作。例如:
var vm = new Vue({
el: '#myForm',
data: {
field1: "",
field2: ""
}
});
$(".date-picker").datepicker({
todayHighlight: true,
autoclose: true,
closeOnDateSelect: true
});
答案 6 :(得分:-1)
使用`而不是&#39;对于模板,因为它包含几行。
这将解决您的问题:)