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 的不确定性,又保证了代码的可靠性和可维护性

相关推荐
这里是杨杨吖4 小时前
SpringBoot+Vue医院预约挂号系统 附带详细运行指导视频
vue.js·spring boot·医院·预约挂号
光影少年6 小时前
vue打包优化方案都有哪些?
前端·javascript·vue.js
木易 士心9 小时前
Ref 和 Reactive 响应式原理剖析与代码实现
前端·javascript·vue.js
被巨款砸中10 小时前
一篇文章讲清Prompt、Agent、MCP、Function Calling
前端·vue.js·人工智能·web
uhakadotcom10 小时前
execjs有哪些常用的api,如何逆向分析网站的加签机制
前端·javascript·面试
闲蛋小超人笑嘻嘻11 小时前
find数组方法详解||Vue3 + uni-app + Wot Design(wd-picker)使用自定义插槽内容写一个下拉选择器
前端·javascript·uni-app
小牛itbull11 小时前
初始化electron项目运行后报错 electron uninstall 解决方法
前端·javascript·electron
rggrgerj12 小时前
Vue3 组件完全指南代码
前端·javascript·vue.js
在逃的吗喽14 小时前
Vue3新变化
前端·javascript·vue.js
Demoncode_y14 小时前
Vue3中基于路由的动态递归菜单组件实现
前端·javascript·vue.js·学习·递归·菜单组件