如何实现一个 autoSize 的 textarea

前言

熟悉 ant-design-vue 的小伙伴应该知道,<a-textarea auto-size /> 可以渲染出一个自适应高度的 textarea(即:高度随着内容的变化而变化);本文将介绍一下这样一个功能是如何实现的。

实现思路

如果我们能够实时的知道 textarea 元素中文本的高度,那么这个功能就很好实现了,只需要实时的将文本的高度赋值给 textarea 元素即可。

但是如何能够实时的知道 textarea 元素中文本的高度呢?这好像并不容易实现,那么我们可以换一种思路,如果有另外一个隐藏起来的 textarea(后文简称 hiddenTextarea), 它的所有尺寸相关 css 属性都与原本的 textarea 保持一致且文本内容也一致,那么这个需求就有可能可以实现;但问题在于 textarea 文本超出高度后会显示滚动条,而不是像其他普通元素一样更新元素的高度(如果是这样,那本文也没必要存在了[手动狗头])。

DOM 中有一个属性叫做 scrollHeight,我们可以获取 hiddenTextareascrollHeight 属性,将它的值赋值给原本 textareaheight 属性,那就实现了这个需求;接下来我们来看看具体如何实现。

实现

  1. 首先需要创建一个 hiddenTextarea,同时保证其尺寸相关的 css 属性都与原本的 textarea 保持一致,我们先来看看有哪些属性是与尺寸相关的:
ts 复制代码
const SIZING_STYLE = [
  'letter-spacing',
  'line-height',
  'padding-top',
  'padding-bottom',
  'font-family',
  'font-weight',
  'font-size',
  'font-variant',
  'text-rendering',
  'text-transform',
  'width',
  'text-indent',
  'padding-left',
  'padding-right',
  'border-width',
  'box-sizing',
  'word-break',
];

只需要将两个 textarea 的这些属性保持一致,那么他们的尺寸就会保持一致。

  1. 通过 window.getComputedStyle 方法获取原本 textarea 的属性值。
ts 复制代码
const style = window.getComputedStyle(textarea)
const sizingStyle = SIZING_STYLE.map(prop => `${prop}: ${style.getPropertyValue(prop)}`).join(';')
  1. 将属性值设置到 hiddenTextarea 上:
ts 复制代码
hiddenTextarea.setAttribute('style', `${sizingStyle};`);
  1. 获取 hiddenTextareascrollHeight 赋值给 textareaheight
ts 复制代码
textarea.style.height = hiddenTextarea.scrollHeight

如此,就实现了一个自适应高度的 textarea

结语

本文主要介绍了如何实现 textarea 自适应高度的原理。

相关推荐
小满zs10 分钟前
Next.js第五章(动态路由)
前端
清沫14 分钟前
VSCode debugger 调试指南
前端·javascript·visual studio code
一颗宁檬不酸1 小时前
页面布局练习
前端·html·页面布局
zhenryx2 小时前
React Native 自定义 ScrollView 滚动条:开箱即用的 IndicatorScrollView(附源码示例)
javascript·react native·react.js·typescript
金木讲编程2 小时前
Claude、Agent与Copilot协作生成Angular应用
前端·ai编程
振华OPPO3 小时前
Vue:“onMounted“ is defined but never used no-unused-vars
前端·javascript·css·vue.js·前端框架
欧雷殿3 小时前
在富阳银湖成立地域化的软件研发团队
前端·程序员·创业
狂炫冰美式4 小时前
前端实时推送 & WebSocket 面试题(2026版)
前端·http·面试
JefferyXZF4 小时前
新手建站零门槛!Vercel+Cloudflare+Namesilo域名购买部署全流程
前端
yinuo4 小时前
微信浏览器缓存机制大揭秘:为什么你总刷不出新页面?
前端