先直接看最终效果展示

直接打开element form界面,将本文最下面的js文件粘贴到console中并执行

刚打开时的界面展示

js中给Java对象赋值 js赋值给input_elementui

console内执行代码后:

js中给Java对象赋值 js赋值给input_前端_02

需求 - element form批量赋值

我们系统(element ui写的),有一个行内编辑的table界面,数据量比较大,并且输入项均为必需。
QA同学提了一个小需求,就是能不能帮她们做一个小工具,可以批量赋值的,可以提高她们的输入效率

初次尝试

这还不简单?N年前就干过。上来就干,先拿input试试

// 初版有问题,只能赋值,无法获取到值
const inputList = document.querySelectorAll('.el-input')
for (const i of inputList) {
  const inpEle = i.querySelector('.el-input__inner')
  inpEle.setAttribute('value', 1)
}

粘贴到console中试一把,结果是界面显示数据设置上去了,但是保存的时候数据获取不到
为啥呢?
因为之前jQuery是从实际的input取值,然后继续下一步;现在是基于vue的开发,数据会存在data中,只操作dom里面的内容,是搞不定的

JS模拟键盘事件实现input输入

不能通过js直接修改dom的值,那咋办?
答案:只能通过js模拟键盘、鼠标事件,通过webapi的方式来调用
普通输入通过InputEvent可以模拟实现

// 普通输入事件
const inputEvt = new InputEvent('input', {
  inputType: 'insertText',
  data: 1,
  dataTransfer: null,
  isComposing: false,
})

// 遍历处理input框
const inputList = document.querySelectorAll('.el-input')
for (const i of inputList) {
  // 在select以及日期框里面的不做处理
  if (i.parentElement.classList.contains('el-select') || i.classList.contains('el-date-editor')) {
    continue
  }
  const inpEle = i.querySelector('.el-input__inner')
  inpEle.value = 1
  inpEle.dispatchEvent(inputEvt)
}

相同操作,实现input type='textarea'输入

这个与input类似,只是由于dom结构的不同,需要写两份代码

// 普通输入事件
const inputEvt = new InputEvent('input', {
  inputType: 'insertText',
  data: 1,
  dataTransfer: null,
  isComposing: false,
})

// 遍历处理textarea框
const textareaList = document.querySelectorAll('.el-textarea')
for (const i of textareaList) {
  const inpEle = i.querySelector('.el-textarea__inner')
  inpEle.value = 1
  inpEle.dispatchEvent(inputEvt)
}

JS模拟鼠标、键盘事件实现select选择

处理完input,还有一个重要的是select
这个需要 MouseEvent + KeyboardEvent 研究了一下select的规律,发现通过三步可以实现
1.获取到select的dom
2.派发点击事件 - 使select的选择列可以出来
3.派发键盘向下按键事件 - 选择第一个option
4.派发回车事件 - 完成select选中

代码如下

// 鼠标点击事情
const mouseEvt = new MouseEvent('click', {
  bubbles: true,
  cancelable: true,
  view: window,
})

// 模拟键盘向下按键
const downEvt = new KeyboardEvent('keydown', {
  bubbles: true, cancelable: true, keyCode: 40,
})

// 模拟键盘回车按键
const enterEvt = new KeyboardEvent('keydown', {
  bubbles: true, cancelable: true, keyCode: 13,
})

/**
 * 遍历处理下拉,模拟 点击-向下按键-回车按键
 */
const selectInputList = document.querySelectorAll('.el-select>.el-input')
for (const inpEle of selectInputList) {
  inpEle.dispatchEvent(mouseEvt)
  inpEle.dispatchEvent(downEvt)
  inpEle.dispatchEvent(enterEvt)
}

js模拟鼠标事件,实现Radio、CheckBox的选择

有上面的input、select的处理经验,处理Radio、CheckBox应当更简单了
思路:遍历获取到Radio/CheckBox的外层元素,然后获取到其第一个节点,派发点击事件即可

// 有问题版本 - 只能触发第一个组件
// 鼠标点击事件
const mouseEvt = new MouseEvent('click', {
  bubbles: true,
  cancelable: true,
  view: window,
})

// 处理radio,默认选中第一个
const radioList = document.querySelectorAll('.el-radio-group')
for (const item of radioList) {
  item.firstElementChild.dispatchEvent(mouseEvt)
}

// 处理checkbox,默认选中第一个
const checkboxList = document.querySelectorAll('.el-checkbox-group')
for (const item of checkboxList) {
  item.firstElementChild.dispatchEvent(mouseEvt)
}

试一把!有问题,第一个可以选中,后面的就不行了
研究了半天,发现了问题,对于Radio、CheckBox来说,MouseEvent不能复用,需要每次使用一个新的
修复版代码

// 每次返回新的鼠标事件
function getMouseEvent() {
  return new MouseEvent('click', {
    bubbles: true,
    cancelable: true,
    view: window,
  })
}

// 处理radio,默认选中第一个
const radioList = document.querySelectorAll('.el-radio-group')
for (const item of radioList) {
  item.firstElementChild.dispatchEvent(getMouseEvent())
}

// 处理checkbox,默认选中第一个
const checkboxList = document.querySelectorAll('.el-checkbox-group')
for (const item of checkboxList) {
  if (!item.firstElementChild.classList.contains('is-checked')) {
    item.firstElementChild.dispatchEvent(getMouseEvent())
  }
}

嗯,这下好使了……

添加上辅助代码,收尾

在界面右下角添加一个按钮,切换界面的时候,点击该按钮也会进行界面操作

function setValue () {
  ...
}

(function () { 
  const btn = document.createElement('input')
  btn.type = 'button'
  btn.style.position = 'fixed'
  btn.style.bottom = '100px'
  btn.style.right = '100px'
  btn.style.zIndex = '9999'
  btn.classList.add('el-button')
  btn.classList.add('el-button--primary')
  btn.value = '批量赋值'
  btn.onclick = function () {
    setValue()
  }
  document.body.appendChild(btn)
})()

最终版本代码地址如下:可以直接贴到console中允许

最终版js地址

前端小工具 - js模拟键盘、鼠标事件给element form批量赋值

总结

目前只是简单的实现了element ui的form赋值,同理,ant design的form赋值,也是类似的操作思想

InputEvent、MouseEvent、KeyboardEvent,在作为webapi,在平常开发中使用的不多,可以通过其mdn的链接查看学习:Web API 接口参考

后面其实可以做的更精细, 比如进一步做成Chrome的插件,实现各种批量赋值的功能