在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 jquery获取随机生成id的服务器控件的三种方法
Jul 11 Javascript
Javascript学习笔记之 函数篇(三) : 闭包和引用
Nov 23 Javascript
JS实现单行文字不间断向上滚动的方法
Jan 29 Javascript
浅谈js基本数据类型和typeof
Aug 09 Javascript
Node.js的环境安装配置(使用nvm方式)
Oct 11 Javascript
JS中关于正则的巧妙操作
Aug 31 Javascript
Chrome调试折腾记之JS断点调试技巧
Sep 11 Javascript
VUE element-ui 写个复用Table组件的示例代码
Nov 18 Javascript
利用node实现一个批量重命名文件的函数
Dec 21 Javascript
微信小程序开发中var that =this的用法详解
Jan 18 Javascript
javascript实现画板功能
Apr 12 Javascript
Vue 请求传公共参数的操作
Jul 31 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
php模拟ping命令(php exec函数的使用方法)
2013/10/25 PHP
php制作中间带自己定义图片二维码的方法
2014/01/27 PHP
Thinkphp 框架扩展之应用模式实现方法分析
2020/04/27 PHP
jquery创建div 实现代码
2009/04/27 Javascript
js 对象是否存在判断
2009/07/15 Javascript
flash 得到自身url参数的代码
2009/11/15 Javascript
Angularjs中UI Router全攻略
2016/01/29 Javascript
解决拦截器对ajax请求的拦截实例详解
2016/12/21 Javascript
微信小程序 在线支付功能的实现
2017/03/14 Javascript
vue.js国际化 vue-i18n插件的使用详解
2017/07/07 Javascript
在vue中获取微信支付code及code被占用问题的解决方法
2019/04/16 Javascript
微信小程序select下拉框实现源码
2019/11/08 Javascript
node使用async_hooks模块进行请求追踪
2021/01/28 Javascript
[02:34]肉山说——泡妞篇
2014/09/16 DOTA
python 正则表达式 概述及常用字符
2009/05/04 Python
python获取本地计算机名字的方法
2015/04/29 Python
Python urllib、urllib2、httplib抓取网页代码实例
2015/05/09 Python
使用pandas把某一列的字符值转换为数字的实例
2019/01/29 Python
python递归法解决棋盘分割问题
2019/07/17 Python
python3实现高效的端口扫描
2019/08/31 Python
Python字符串和正则表达式中的反斜杠('\')问题详解
2019/09/03 Python
将tensorflow.Variable中的某些元素取出组成一个新的矩阵示例
2020/01/04 Python
解决tensorflow由于未初始化变量而导致的错误问题
2020/01/06 Python
TensorFlow2.0矩阵与向量的加减乘实例
2020/02/07 Python
keras 解决加载lstm+crf模型出错的问题
2020/06/10 Python
scrapy-redis分布式爬虫的搭建过程(理论篇)
2020/09/29 Python
Python爬虫之Selenium设置元素等待的方法
2020/12/04 Python
python实现按日期归档文件
2021/01/30 Python
解决Pycharm 运行后没有输出的问题
2021/02/05 Python
德国著名廉价网上药店:Shop-Apotheke
2017/07/23 全球购物
十八届三中全会感言
2014/03/10 职场文书
政风行风评议整改方案
2014/09/15 职场文书
无犯罪记录证明范本
2014/09/15 职场文书
大学生个人学年总结
2015/02/15 职场文书
2015年宣传工作总结
2015/04/08 职场文书
汶川大地震感悟
2015/08/10 职场文书