Implementation of textarea adaptive height solution in Vue

Implementation of textarea adaptive height solution in Vue

Give the solution first. Students who need Vue stack can directly download vue-awesome-textarea

Hidden Problems

Apart from native JS, most of the UI libraries of the framework support the adaptive textarea height function, but one function is generally overlooked, which is the adaptive height echo.

When using these libraries, we can easily type content in the textarea, and it will automatically extend one line when it exceeds the range to ensure the content is highly adaptive. When we submit content and use the same UI to render on other pages, trouble arises. Some UI libraries do not support adaptive echo, which requires us to calculate a base value between line height, number of lines, and even height to achieve echo.

Solution to adaptive height

There are two common solutions. One is to add a ghost DOM in the "remote area" of the page to simulate input line breaks. This DOM may be a div with the editable attribute set to true or an identical textarea.
Taking the input component of element-ui as an example, when we enter a value in the component, the resizeTextarea method will be called

resizeTextarea() {
 if (this.$isServer) return;
 const { autosize, type } = this;
 if (type !== 'textarea') return;
 if (!autosize) {
  this.textareaCalcStyle = {
   minHeight: calcTextareaHeight(this.$refs.textarea).minHeight
  };
  return;
 }
 const minRows = autosize.minRows;
 const maxRows = autosize.maxRows;

 this.textareaCalcStyle = calcTextareaHeight(this.$refs.textarea, minRows, maxRows);
}

When autosize is set to true, the textarea is set to adaptive height. At this time, the height of the textarea will be calculated in real time through the calcTextareaHeight method.

 export default function calcTextareaHeight(
 targetElement,
 minRows = 1,
 maxRows = null
) {
 if (!hiddenTextarea) {
  hiddenTextarea = document.createElement('textarea');
  document.body.appendChild(hiddenTextarea);
 }

 let {
  paddingSize,
  borderSize,
  boxSizing,
  contextStyle
 } = calculateNodeStyling(targetElement);

 hiddenTextarea.setAttribute('style', `${contextStyle};${HIDDEN_STYLE}`);
 hiddenTextarea.value = targetElement.value || targetElement.placeholder || '';

 let height = hiddenTextarea.scrollHeight;
 const result = {};

 if (boxSizing === 'border-box') {
  height = height + borderSize;
 } else if (boxSizing === 'content-box') {
  height = height - paddingSize;
 }

 hiddenTextarea.value = '';
 let singleRowHeight = hiddenTextarea.scrollHeight - paddingSize;

 if (minRows !== null) {
  let minHeight = singleRowHeight * minRows;
  if (boxSizing === 'border-box') {
   minHeight = minHeight + paddingSize + borderSize;
  }
  height = Math.max(minHeight, height);
  result.minHeight = `${minHeight}px`;
 }
 if (maxRows !== null) {
  let maxHeight = singleRowHeight * maxRows;
  if (boxSizing === 'border-box') {
   maxHeight = maxHeight + paddingSize + borderSize;
  }
  height = Math.min(maxHeight, height);
 }
 result.height = `${ height }px`;
 hiddenTextarea.parentNode && hiddenTextarea.parentNode.removeChild(hiddenTextarea);
 hiddenTextarea = null;
 return result;
};

We can see

 if (!hiddenTextarea) {
  hiddenTextarea = document.createElement('textarea');
  document.body.appendChild(hiddenTextarea);
}

Element-ui creates a textarea DOM and copies the style of the real textarea to the hiddenTextarea through the calculateNodeStyling method (the overflow is not synchronized, the real textarea is hidden). Then monitor the input value of textarea and synchronize it to hiddenTextarea. At the same time, synchronize the scrollHeight of hiddenTextarea to the height of textarea, and finally destroy the dom.

Regarding style synchronization, element uses the two APIs getComputedStyle and getPropertyValue. Of course, if you wrap it yourself, you can also use the CSS preprocessor mixin.

The second solution is similar to the first solution, but no additional DOM is created. Take the vue-awesome-textarea at the beginning as an example:

 init() {
  this.initAutoResize()
},
initAutoResize () {
  this.autoResize && this.$nextTick(this.calcResize)
}

When the page is mounted or the content changes and autoResize is turned on, the this.calcResize method is executed.

 calcResize() {
 this.resetHeight()
 this.calcTextareaH()
},

resetHeight() {
 this.height = 'auto'
},

calcTextareaH() {
 let contentHeight = this.calcContentHeight()
 this.height = this.calcHeightChange(contentHeight) + 'px'
 if (this.needUpdateRows(contentHeight)) {
  this.updateRows(contentHeight)
 }
 this.oldContentHeight = contentHeight
},

calcContentHeight () {
 const { paddingSize } = this.calcNodeStyle(this.$el)
 return this.$el.scrollHeight - paddingSize
},

resetHeight() is used to initialize the height of the textarea, the default is auto. The calcTextareaH() method is used to calculate the height of the content area (the scrollHeight of the textarea minus the height of the padding), and synchronize the calculated height to the height of the textarea in real time:
this.height = this.calcHeightChange(contentHeight) + 'px'

Compared with Solution 1, this solution adopts the same idea (dynamically modifying the height), but reduces the additional DOM creation and destruction process.
In addition, vue-awesome-textarea also supports callback of the number of lines during the adaptive process, which can better support data echo. The implementation method is also very simple:

 computed: {
 ...
 oneRowsHeight() {
  return this.calcContentHeight() / Number(this.rows) || 0
 }
 ...
}

In computed we calculate the height of a single line, and when executing the this.calcTextareaH() method we record the content height:

 this.oldContentHeight = contentHeight

Next, we will check whether there is an added row operation. If it is added, the new content height will be different from the old content height:

 needUpdateRows(newContentHeight) {
  return this.oldContentHeight !== newContentHeight
},

At this point we will emit the latest row height outside the component:

 updateRows(contentHeight) {
  this.$emit('getRows', Math.round(contentHeight / this.oneRowsHeight))
}

This is the end of this article about the implementation of textarea adaptive height solution in Vue. For more relevant Vue textarea adaptive content, please search 123WORDPRESS.COM's previous articles or continue to browse the following related articles. I hope everyone will support 123WORDPRESS.COM in the future!

You may also be interested in:
  • Detailed explanation of the idea of ​​Vue to implement fixed number of input lines and add underline style for textarea
  • Vue implements dynamic display of the remaining words in textarea

<<:  Text pop-up effects implemented with CSS3

>>:  Detailed explanation of some settings for Table adaptation and overflow

Recommend

Detailed tutorial on installing Docker and docker-compose suite on Windows

Table of contents Introduction Download and insta...

A colorful cat under Linux

Friends who have used the Linux system must have ...

JavaScript canvas to achieve scratch lottery example

This article shares the specific code of JavaScri...

How to delete a MySQL table

It is very easy to delete a table in MySQL, but y...

How to Install Xrdp Server (Remote Desktop) on Ubuntu 20.04

Xrdp is an open source implementation of Microsof...

Analyze MySQL replication and tuning principles and methods

1. Introduction MySQL comes with a replication so...

Understanding and application scenarios of enumeration types in TypeScript

Table of contents 1. What is 2. Use Numeric Enume...

Explore how an LED can get you started with the Linux kernel

Table of contents Preface LED Trigger Start explo...

Move MySQL database to another disk under Windows

Preface Today I installed MySQL and found that th...

Summary of common Nginx techniques and examples

1. Priority of multiple servers For example, if e...

How to load Flash in HTML (2 implementation methods)

First method : CSS code: Copy code The code is as ...