Vue3 + TypeScript + Element Plus + el-input 输入框列表按回车聚焦到下一行

应用效果:从第一行输入1,按回车,聚焦到第二行输入2,按回车,聚焦到第三行......

一、通过元素 id,聚焦到下一行的输入框

关键技术点:

1、动态设置元素 id 属性为::id="`input-apply-amount-${(option as IReagentOption).id}`"

2、设置回车监听:@keyup.enter.native="onEnterPressDown((option as IReagentOption).id)"

html 复制代码
                <el-input
                  ......
                  :id="`input-apply-amount-${(option as IReagentOption).id}`"
                  @keyup.enter.native="onEnterPressDown((option as IReagentOption).id)" />

3、通过 document.getElementById 获取到指定元素

4、focus 和 select,聚焦、全选

TypeScript 复制代码
// 通过元素 id,聚焦到下一行的输入框
const focusNextRowByElementId = (objId: number) => {
  // 通过 objId 获取当前行索引
  let currentRowIndex = selectedOptionIds.value.indexOf(objId);
  // 下一行的输入框
  let nextInput: HTMLInputElement;
  if (currentRowIndex + 1 < selectedOptionIds.value.length) {
    // 下一行索引的 objId
    let nextRowObjId = selectedOptionIds.value[currentRowIndex + 1];
    nextInput = document.getElementById(`input-apply-amount-${nextRowObjId}`) as HTMLInputElement;
  } else {
    // 最后一行聚焦到申领用途输入框
    nextInput = document.getElementById("transfer-right-footer-purpose-input") as HTMLInputElement;
  }
  nextInput?.focus();
  nextInput?.select();
};

二、通过元素 ref,聚焦到下一行的输入框

关键技术点:

1、动态设置元素 ref 属性为::ref="`input-apply-amount-${(option as IReagentOption).id}`"

2、设置回车监听:@keyup.enter.native="onEnterPressDown((option as IReagentOption).id)"

html 复制代码
                <el-input
                  ......
                  :ref="`input-apply-amount-${(option as IReagentOption).id}`"
                  @keyup.enter.native="onEnterPressDown((option as IReagentOption).id)" />

3、通过 getCurrentInstance() 获取当前组件实例,再通过 refs 获取到元素列表

4、focus 和 select,聚焦、全选

TypeScript 复制代码
<script setup lang="ts" name="ReagentApplyDialog">
......
import { getCurrentInstance } from "vue";

// 当前组件实例,相当 vue2 的this
const thisInstance = getCurrentInstance();

// 通过元素 ref,聚焦到下一行的输入框
const focusNextRowByElementRef = (objId: number) => {
  if (!thisInstance) return;

  // 获取输入框实例 Record
  let refs = thisInstance.refs as Record<string, HTMLInputElement>;

  // 通过 objId 获取当前行索引
  let currentRowIndex = selectedOptionIds.value.indexOf(objId);
  if (currentRowIndex + 1 < selectedOptionIds.value.length) {
    // 下一行索引的 objId
    let nextRowObjId = selectedOptionIds.value[currentRowIndex + 1];
    // 聚焦到下一行的输入框
    refs[`input-apply-amount-${nextRowObjId}`].focus();
    refs[`input-apply-amount-${nextRowObjId}`].select();
  } else {
    // 最后一行聚焦到申领用途输入框
    refs["transfer-right-footer-purpose-input"].focus();
    refs["transfer-right-footer-purpose-input"].select();
  }
};
......
</script>

实例完整代码:

html 复制代码
<script setup lang="ts" name="ReagentApplyDialog">
......
import { getCurrentInstance, nextTick, onMounted, ref, watch } from "vue";

// 当前组件实例,相当 vue2 的this
const thisInstance = getCurrentInstance();

// 按回车,申领数量输入框按回车
const onEnterPressDown = (objId: number) => {
  // 奇数
  if (objId % 2 === 1) {
    // 通过元素 id,聚焦到下一行的输入框
    focusNextRowByElementId(objId);
  }
  // 偶数
  else {
    // 通过元素 ref,聚焦到下一行的输入框
    focusNextRowByElementRef(objId);
  }
};

// 通过元素 id,聚焦到下一行的输入框
const focusNextRowByElementId = (objId: number) => {
  // 通过 objId 获取当前行索引
  let currentRowIndex = selectedOptionIds.value.indexOf(objId);
  // 下一行的输入框
  let nextInput: HTMLInputElement;
  if (currentRowIndex + 1 < selectedOptionIds.value.length) {
    // 下一行索引的 objId
    let nextRowObjId = selectedOptionIds.value[currentRowIndex + 1];
    nextInput = document.getElementById(`input-apply-amount-${nextRowObjId}`) as HTMLInputElement;
  } else {
    // 最后一行聚焦到申领用途输入框
    nextInput = document.getElementById("transfer-right-footer-purpose-input") as HTMLInputElement;
  }
  nextInput?.focus();
  nextInput?.select();
};

// 通过元素 ref,聚焦到下一行的输入框
const focusNextRowByElementRef = (objId: number) => {
  if (!thisInstance) return;

  // 获取输入框实例 Record
  let refs = thisInstance.refs as Record<string, HTMLInputElement>;

  // 通过 objId 获取当前行索引
  let currentRowIndex = selectedOptionIds.value.indexOf(objId);
  if (currentRowIndex + 1 < selectedOptionIds.value.length) {
    // 下一行索引的 objId
    let nextRowObjId = selectedOptionIds.value[currentRowIndex + 1];
    // 聚焦到下一行的输入框
    refs[`input-apply-amount-${nextRowObjId}`].focus();
    refs[`input-apply-amount-${nextRowObjId}`].select();
  } else {
    // 最后一行聚焦到申领用途输入框
    refs["transfer-right-footer-purpose-input"].focus();
    refs["transfer-right-footer-purpose-input"].select();
  }
};
......
</script>

<template>
......
        <el-transfer
          ......>
          <!-- 自定义列表数据项的内容 -->
          <template #default="{ option }">
                ......
                <el-input
                  v-if="selectedOptionIds.includes((option as IReagentOption).id)"
                  :ref="`input-apply-amount-${(option as IReagentOption).id}`"
                  :id="`input-apply-amount-${(option as IReagentOption).id}`"
                  class="input-apply-amount"
                  style="width: 85px; text-align: center"
                  v-model="(option as IReagentOption).applyAmount"
                  placeholder="输入申领数量"
                  size="small"
                  clearable
                  @input="(option as IReagentOption).applyAmount = Number(formatToNumber($event, 0))"
                  @keyup.enter.native="onEnterPressDown((option as IReagentOption).id)" />
                ......
          </template>
        </el-transfer>
......
</template>
相关推荐
white.tie6 分钟前
一个手机请求头的随机库
开发语言·javascript·python
不像程序员的程序媛11 分钟前
es按时间 小时聚合
java·前端·elasticsearch
bobz96516 分钟前
为什么 不能在浏览器里直接用 Node.js 专用模块?
前端
萌萌哒草头将军26 分钟前
⚡️⚡️⚡️ 开源了!原来 Vite 加载图片还可以这样啊!🚀🚀🚀
javascript·vue.js·react.js
断竿散人27 分钟前
📏CSS尺寸单位终极手册:从px到fr的完全征服指南
前端·css·响应式设计
Rrvive30 分钟前
Python学习笔记
前端
Aisanyi30 分钟前
【鸿蒙开发】适配深浅色模式
前端·harmonyos
onejason30 分钟前
如何利用 PHP 爬虫按关键字搜索 Amazon 商品
前端·后端·php
咚咚咚ddd30 分钟前
前端基建:使用plus api实现app通知权限管理
前端·javascript·前端工程化
懒猫爱上鱼32 分钟前
Kotlin协程简单介绍
前端