在Vant的基础上封装下拉日期控件的代码示例


Posted in Javascript onDecember 05, 2018

需求分析

在实际项目中,表单里面的日期选择是常用的组件。Vant有提供日期组件,但是居然没有提供下拉形式的日期组件,不过该有的元件都有,就自己封装一个。

封装组件过程中我们要解决:

  • 和表单的样式能兼容
  • 错误提示
  • 参数问题
  • 事件机制
  • 格式化

解决问题

就给新的组件取名为 VantFieldDate

期望使用的时候是这样的

<vant-field-date
 label="发布时间"
 v-model="formData.publishDate"
 type="datetime"
 :max-date="new Date()"
/>

具体实现,我贴上代码详细讲解。

<template>
 <div class="vant-field-date">
  <van-cell
   :title="label"
   :class="{'readonly': readonly, 'placeholder' : text}"
   :is-link="!readonly"
   :required="required"
   @click="show">
   <!-- 显示当前值,没有值显示提示文字 -->
   {{ text ? text : placeholder }}
   <!-- 自定义错误显示 -->
   <div
    v-if="$attrs.error"
    v-text="$attrs['error-message']"
    class="van-field__error-message"
   />
  </van-cell>
  <!-- 用 actionsheet 来包裹弹出层日期控件 -->
  <van-actionsheet v-model="isShowPicker">
   <!-- $attrs 可以把根节点的attr放到目标组件上,如此可以像使用 DatePicker 组件一样使用这个新组件 -->
   <van-datetime-picker
    v-bind="$attrs"
    :type="type"
    title="请选择日期"
    :min-date="minDate"
    :max-date="maxDate"
    @cancel="cancel"
    @confirm="confirm"
   />
  </van-actionsheet>
 </div>
</template>

<script>
 export default {
  name: 'VantFieldDate',
  inheritAttrs: false, // https://cn.vuejs.org/v2/api/#inheritAttrs
  props: {
   value: {
    type: [Number, Date],
    default: undefined // 值不能是 null,DatePicker会报错
   },
   // Cell 显示的文字
   label: {
    type: String,
    default: null
   },
   // 必填的星号
   required: {
    type: Boolean,
    default: false
   },
   // 只读状态
   readonly: {
    type: Boolean,
    default: false
   },
   // 占位提示文字
   placeholder: {
    type: String,
    default: '请选择'
   },
   // 展示的格式化
   format: {
    type: String,
    default: null
   }
  },
  data() {
   return {
    selectedItem: null,
    isShowPicker: false
   }
  },
  computed: {
   // 展示的格式化,时间提交的值是Date类型数据
   formatFormula() {
    if(this.format){
     return this.format
    } else if (this.type === 'date') {
     return 'yyyy-MM-dd'
    } else if (this.type === 'datetime') {
     return 'yyyy-MM-dd hh:mm'
    } else if (this.type === 'time') {
     return 'hh:mm'
    } else if (this.type === 'year-month') {
     return 'yyyy-MM'
    }
   },
   text() {
    return this.value ? this.dateFormat(this.value, this.formatFormula) : ''
   }
  },
  methods: {
   dateFormat: (value, format) => {
    if (!value) return
    if (!(value instanceof Date)) {
     value = new Date(value)
    }
    let o = {
     'M+': value.getMonth() + 1, // month
     'd+': value.getDate(), // day
     'h+': value.getHours(), // hour
     'm+': value.getMinutes(), // minute
     's+': value.getSeconds(), // second
     'q+': Math.floor((value.getMonth() + 3) / 3), // quarter
     'S': value.getMilliseconds() // millisecond
    }

    if (!format || format === '') {
     format = 'yyyy-MM-dd hh:mm:ss'
    }

    if (/(y+)/.test(format)) {
     format = format.replace(RegExp.$1, (value.getFullYear() + '').substr(4 - RegExp.$1.length))
    }

    for (let k in o) {
     if (new RegExp('(' + k + ')').test(format)) {
      format = format.replace(RegExp.$1, RegExp.$1.length === 1 ? o[k] : ('00' + o[k]).substr(('' + o[k]).length))
     }
    }
    return format
   },
   show() {
    if (!this.readonly) {
     this.isShowPicker = true
    }
   },
   confirm(value) {
    // 更新 v-model 绑定的 value 值,第二个参数是毫秒数,第三个参数是原始值,根据自己的项目的数据结构来修改
    // input 事件同时也会触发 vee-validate 的验证事件
    this.$emit('input', value.getTime(), value)
    // onChange事件,虽然重写 @input可以实现,但这样会破坏 v-model 写法。
    this.$emit('change', value.getTime(), value)
    this.cancel()
   },
   // 隐藏弹框
   cancel() {
    this.isShowPicker = false
   }
  }
 }
</script>

效果

在Vant的基础上封装下拉日期控件的代码示例

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
js异或加解密效果代码
Jun 25 Javascript
javascript操作cookie_获取与修改代码
May 21 Javascript
jquery获取下拉列表的值为null的解决方法
Mar 18 Javascript
jquery多行滚动/向左或向上滚动/响应鼠标实现思路及代码
Jan 23 Javascript
CSS+JS实现点击文字弹出定时自动关闭DIV层菜单的方法
May 12 Javascript
jQuery实现带玻璃流光质感的手风琴特效
Nov 20 Javascript
谈谈PHP中相对路径的问题与绝对路径的使用
Aug 16 Javascript
js将字符串中的每一个单词的首字母变为大写其余均为小写
Jan 05 Javascript
深入理解AngularJs-scope的脏检查(一)
Jun 19 Javascript
React Native使用Modal自定义分享界面的示例代码
Oct 31 Javascript
VueRouter导航守卫用法详解
Dec 25 Javascript
vue中echarts图表大小适应窗口大小且不需要刷新案例
Jul 19 Javascript
Vant的安装和配合引入Vue.js项目里的方法步骤
Dec 05 #Javascript
微信小程序实现图片滚动效果示例
Dec 05 #Javascript
详解vue中的computed的this指向问题
Dec 05 #Javascript
使用rollup打包JS的方法步骤
Dec 05 #Javascript
微信小程序入门之广告条实现方法示例
Dec 05 #Javascript
Vue实现本地购物车功能
Dec 05 #Javascript
node和vue实现商城用户地址模块
Dec 05 #Javascript
You might like
使用cookie实现统计访问者登陆次数
2013/06/08 PHP
PHP设计模式之命令模式的深入解析
2013/06/13 PHP
PHP页面跳转操作实例分析(header方法)
2016/09/28 PHP
JQuery UI皮肤定制
2009/07/27 Javascript
JavaScript DOM学习第六章 表单实例
2010/02/19 Javascript
javascript:history.go()和History.back()的区别及应用
2012/11/25 Javascript
把jQuery的类、插件封装成seajs的模块的方法
2014/03/12 Javascript
javascript父、子页面交互技巧总结
2014/08/08 Javascript
node.js中的fs.link方法使用说明
2014/12/15 Javascript
JavaScript实现更改网页背景与字体颜色的方法
2015/02/02 Javascript
基于jQuery创建鼠标悬停效果的方法
2015/03/07 Javascript
HTML的select控件美化
2017/03/27 Javascript
javascript trie前缀树的示例
2018/01/29 Javascript
vue-cli 首屏加载优化问题
2018/11/06 Javascript
微信小程序实现简单评论功能
2018/11/28 Javascript
使用Jenkins部署React项目的方法步骤
2019/03/11 Javascript
微信小程序 冒泡事件原理解析
2019/09/27 Javascript
详解如何在JS代码中消灭for循环
2019/12/11 Javascript
JS实现鼠标按下拖拽效果
2020/07/23 Javascript
vue+ElementUI 关闭对话框清空验证,清除form表单的操作
2020/08/06 Javascript
在 Django/Flask 开发服务器上使用 HTTPS
2014/07/03 Python
Python操作RabbitMQ服务器实现消息队列的路由功能
2016/06/29 Python
Python递归实现汉诺塔算法示例
2018/03/19 Python
Python logging模块用法示例
2018/08/28 Python
django使用admin站点上传图片的实例
2019/07/28 Python
python 基于UDP协议套接字通信的实现
2021/01/22 Python
金属材料工程毕业生个人的自我评价
2013/11/28 职场文书
企业宣传方案
2014/03/04 职场文书
农村改厕实施方案
2014/03/22 职场文书
2014年班组建设工作总结
2014/12/01 职场文书
公司股份合作协议书
2014/12/07 职场文书
学校食品安全责任书
2015/01/29 职场文书
《叶问2》观后感
2015/06/15 职场文书
win10安装配置nginx的过程
2021/03/31 Servers
万能密码的SQL注入漏洞其PHP环境搭建及防御手段
2021/09/04 SQL Server
python数字类型和占位符详情
2022/03/13 Python