vue3:触发自动el-input输入框焦点

问:

怎么自动触发el-input输入框焦点

回答:

方法一:使用 nextTick + 组件暴露的 focus 方法(推荐)

复制代码
import { nextTick } from 'vue';

onMounted(async () => {
  await nextTick(); // 等待DOM更新完成
  
  if (searchInputRef.value) {
    searchInputRef.value.focus();
  }
});

方法二:使用 watch + nextTick(更保险)

复制代码
import { watch, nextTick } from 'vue';

// 监听组件是否已经挂载并准备好
watch(
  () => searchInputRef.value,
  (newVal) => {
    if (newVal) {
      nextTick(() => {
        newVal.focus();
      });
    }
  },
  { immediate: true } // 立即执行一次
);

方法三:使用自定义指令(最优雅)

复制代码
// 全局或局部注册指令
const vFocus = {
  mounted(el, binding) {
    const input = el.querySelector('input') || el.querySelector('.el-input__inner');
    if (input) {
      // 使用 requestAnimationFrame 确保在浏览器渲染周期内执行
      requestAnimationFrame(() => {
        input.focus();
      });
    }
  }
};

// 在模板中使用
<el-input
  v-focus
  ref="searchInputRef"
  // ...其他属性
>
</el-input>

方法四:使用 Intersection Observer(适用于需要滚动的场景)

复制代码
onMounted(() => {
  if (!searchInputRef.value) return;

  const observer = new IntersectionObserver((entries) => {
    entries.forEach(entry => {
      if (entry.isIntersecting) {
        nextTick(() => {
          searchInputRef.value.focus();
        });
        observer.disconnect(); // 执行一次后断开观察
      }
    });
  });

  observer.observe(searchInputRef.value.$el);
});

方法五:组合式函数(最可复用)

复制代码
// 创建 useAutoFocus.js
import { ref, onMounted, nextTick } from 'vue';

export function useAutoFocus() {
  const targetRef = ref(null);
  
  const focus = async () => {
    await nextTick();
    if (targetRef.value) {
      targetRef.value.focus();
    }
  };

  onMounted(focus);

  return {
    targetRef,
    focus
  };
}

// 在组件中使用
import { useAutoFocus } from './useAutoFocus';

const { targetRef: searchInputRef } = useAutoFocus();

方法六:使用 Promise 和 requestAnimationFrame(最保险)

复制代码
onMounted(() => {
  const focusInput = () => {
    return new Promise((resolve) => {
      requestAnimationFrame(() => {
        if (searchInputRef.value) {
          searchInputRef.value.focus();
          resolve(true);
        } else {
          resolve(false);
        }
      });
    });
  };

  // 尝试多次,直到成功
  const tryFocus = async (maxAttempts = 5) => {
    for (let i = 0; i < maxAttempts; i++) {
      const success = await focusInput();
      if (success) break;
      await new Promise(resolve => setTimeout(resolve, 50)); // 短暂延迟
    }
  };

  tryFocus();
});

推荐方案

最推荐使用第一种方法(nextTick),因为:

  1. 官方推荐 :Vue 内置的 nextTick 是处理DOM更新后操作的标准方式

  2. 性能更好 :比 setTimeout 更高效

  3. 更可靠:确保在组件完全渲染后才执行焦点设置

  4. 代码简洁:不需要猜测延迟时间

复制代码
// 最佳实践
onMounted(async () => {
  await nextTick();
  searchInputRef.value?.focus();
});

这样既避免了 setTimeout 的不确定性,又保证了代码的可靠性和可维护性

相关推荐
小桥风满袖2 小时前
极简三分钟ES6 - 迭代器Iterator
前端·javascript
小菜全2 小时前
ElementUI 组件概览
前端·vue.js·elementui
Developer-YC2 小时前
像素图生成小程序开发全解析:从图片上传到Excel图纸
java·javascript·图像处理·微信小程序·excel
lichong9512 小时前
【混合开发】vue+Android、iPhone、鸿蒙、win、macOS、Linux之android 把assert里的dist.zip 包解压到sd卡里
android·vue.js·iphone
前端小巷子3 小时前
原生 JS 打造三级联动
前端·javascript·面试
江城开朗的豌豆3 小时前
useEffect vs componentDidUpdate:谁才是真正的更新之王?
前端·javascript·react.js
江城开朗的豌豆3 小时前
解密useEffect:让副作用无所遁形!
前端·javascript·react.js
用户6120414922133 小时前
springmvc做的学生考勤管理系统
javascript·后端·spring
rookie fish4 小时前
Electron+Vite+Vue项目中,如何监听Electron的修改实现和Vue一样的热更新?[特殊字符]
前端·vue.js·electron