Expandir automaticamente uma área de texto usando jQuery


128

Como posso fazer com que uma área de texto se expanda automaticamente usando o jQuery?

Eu tenho uma caixa de texto para explicar a agenda da reunião, então quero expandi-la quando o texto da minha agenda continuar crescendo nessa área.


A área de texto é autoexpansível.
Andrew Sullivan


Por favor, considere mudar a sua pergunta e remover a palavra "jQuery"
vsync

Respostas:


113

Eu tentei muitos e este é ótimo. Link está morto. A versão mais recente está disponível aqui . Veja abaixo a versão antiga.
Você pode tentar pressionar e segurar a tecla Enter na área de texto. Compare o efeito com o outro plug-in de expansão automática de área de texto ....

editar com base no comentário

$(function() {
   $('#txtMeetingAgenda').autogrow();
});

nota: você deve incluir os arquivos js necessários ...

Para evitar que a barra de rolagem na textarea de piscar e desligar durante a expansão / contração, você pode definir o overflowque hiddenbem:

$('#textMeetingAgenda').css('overflow', 'hidden').autogrow()




Atualizar:

O link acima está quebrado. Mas você ainda pode obter os arquivos javascript aqui .


se o meu id caixa de texto é txtMeetingAgenda então como posso implementar Auto textArea propriedade em jquery
Piyush

aqui clique evento é definir ou não
Piyush

trata-se de área de texto, mas e quanto a textbox.is esse plugin também funciona para caixa de texto?
Piyush

1
esse plug-in não funciona bem, tente
redimensionar automaticamente

Funciona bem apenas de uma maneira quando a área de texto está crescendo. Mas quando você começa a excluir o texto, isso não reduz automaticamente a altura.
Ekashking

166

Se você não quer um plugin, existe uma solução muito simples

$("textarea").keyup(function(e) {
    while($(this).outerHeight() < this.scrollHeight + parseFloat($(this).css("borderTopWidth")) + parseFloat($(this).css("borderBottomWidth"))) {
        $(this).height($(this).height()+1);
    };
});

Veja-o trabalhando em um jsFiddle que eu costumava responder a outra pergunta da área de texto aqui .

Para responder à pergunta de fazer isso ao contrário ou diminuí-lo à medida que o texto é removido: jsFiddle

E se você quiser um plugin

@Jason projetou um aqui


1
Eu tive que remover + parseFloat ($ (this) .css ("borderTopWidth")) + parseFloat ($ (this) .css ("borderBottomWidth")), caso contrário, ele executa loops infinitamente com uma área de texto estilizada com bootstrap
blacelle

3
Versão jQuery-less:if (this.clientHeight < this.scrollHeight) { this.style.height = this.scrollHeight + 'px'; }
Georgii Ivankin 27/07/2015

3
@metakungfu Bem, em vez de apenas votar, talvez você possa ajudar a melhorar a resposta. Eu não tenho nenhuma maneira de testar o atm da Apple e o safari para Windows foi descontinuado há muito tempo. Então, eu realmente não posso fazer nada para determinar por que não funcionaria. Em termos de código, não há nada errado com esta solução. Você tem certeza de que é o código que está quebrando e não o safari que quebra no jsfiddle? o safari é bastante exigente e o violino passou por uma grande atualização. Mayeb você pode abrir dev con e fornecer mais alguns detalhes?
SpYk3HH 21/07

1
também precisa textarea conjunto overflow-y para oculto para evitar piscar de rolagem
iulial

2
Se eu quiser remover as linhas adicionadas, o tamanho ampliado da caixa de texto permanecerá o mesmo. É possível que ele também possa ser redimensionado automaticamente novamente?
Steven

33

Aumenta / diminui a área de texto. Esta demonstração utiliza jQuery para ligação de eventos, mas não é uma obrigação de forma alguma.
( sem suporte do IE - o IE não responde à alteração do atributo de linhas )

PÁGINA DE DEMONSTRAÇÃO


HTML

<textarea class='autoExpand' rows='3' data-min-rows='3' placeholder='Auto-Expanding Textarea'></textarea>

CSS

textarea{  
  display:block;
  box-sizing: padding-box;
  overflow:hidden;

  padding:10px;
  width:250px;
  font-size:14px;
  margin:50px auto;
  border-radius:8px;
  border:6px solid #556677;
}

javascript (atualizado)

$(document)
    .one('focus.textarea', '.autoExpand', function(){
        var savedValue = this.value;
        this.value = '';
        this.baseScrollHeight = this.scrollHeight;
        this.value = savedValue;
    })
    .on('input.textarea', '.autoExpand', function(){
        var minRows = this.getAttribute('data-min-rows')|0,
            rows;
        this.rows = minRows;
        rows = Math.ceil((this.scrollHeight - this.baseScrollHeight) / 16);
        this.rows = minRows + rows;
    });

2
porque o código deve esperar um pouco para a letra digitada a ser impresso no interior do textarea e por isso a mudança dimensões do textarea
vsync

1
Entendo, a solução está quebrada quando o texto é copiado (CTRL + V) de qualquer fonte. Embora, se o texto for digitado manualmente - isso é incrível e muito suave.
Satya Kalluri 27/08/14

1
@ vsync: como você modificaria para várias áreas de texto? Ou seja, você tem textarea1 e textarea2 e deseja permitir que ambos cresçam?
jmoreno

1
Essa deve ser a melhor resposta! Eu postei uma pergunta com base em que, se alguém estiver interessado: stackoverflow.com/questions/29930300/...
gsamaras

1
@AllyMurray - Fui agora e atualizei para 16, pois parece dar melhor resultado. e sim, essa é a altura esperada da linha
vsync


20

Você pode tentar este

$('#content').on('change keyup keydown paste cut', 'textarea', function () {
        $(this).height(0).height(this.scrollHeight);
    }).find('textarea').change();
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>
<div id="content">
  <textarea>How about it</textarea><br />
  <textarea rows="5">111111
222222
333333
444444
555555
666666</textarea>
</div>


se você deseja obter o scrollHeight de $ (this), você pode usar $ (this) .prop ('scrollHeight'); Em vez disso;)
Samuel Vicent

3
pisca a cada quebra de linha
joe

11

Graças ao SpYk3HH, comecei com a solução dele e a transformei nessa solução, que adiciona a funcionalidade de redução e é ainda mais simples e rápida, presumo.

$("textarea").keyup(function(e) {
    $(this).height(30);
    $(this).height(this.scrollHeight + parseFloat($(this).css("borderTopWidth")) + parseFloat($(this).css("borderBottomWidth")));
});

Testado no navegador atual Chrome, Firefox e Android 2.3.3.

Você pode ver os flashes das barras de rolagem em alguns navegadores. Adicione este CSS para resolver isso.

textarea{ overflow:hidden; }

não funciona nada bem: / todo caractere digitado, adiciona altura.
vsync 16/10

1
Funciona perfeito para mim! Definitivamente, não há nada no código simples para fazer o que você diz que faz.
Drolex 26/10/2013

1
Sim, eu vejo agora, seu código deve ter a linha que define a altura para 30 inicialmente. esta é provavelmente uma ilne necessária-un, porque obriga o desenvolvedor para ajustar o código, em vez do código de compreender e adaptar-se ao textarea,
vsync

1
Isso está incorreto. O desenvolvedor não precisa ajustar o código. É necessário definir a altura para 30 e funciona para tudo. O usuário nunca vê isso acontecer. É assim que é possível reduzir a área de texto. O código entende e se adapta à área de texto. Esse é ponto principal.
Drolex

Você está certo. página de demonstração: jsbin.com/ObEcoza/2/edit (definir a altura para 1pxparecer melhor)
vsync

10

Para definir uma área de texto expansível automaticamente, é necessário fazer duas coisas:

  1. Expanda -o quando você clicar na tecla Enter dentro dele ou digite o conteúdo mais de uma linha.
  2. E reduza -o em desfoque para obter o tamanho real se o usuário tiver inserido espaços em branco. ( Bônus )

Aqui está uma função artesanal para realizar a tarefa.

Funcionando bem com quase todos os navegadores (<IE7) . Aqui está o método:

    //Here is an event to get TextArea expand when you press Enter Key in it.
    // intiate a keypress event
    $('textarea').keypress(function (e) {  
       if(e.which == 13)   {   
       var control = e.target;                     
       var controlHeight = $(control).height();          
      //add some height to existing height of control, I chose 17 as my line-height was 17 for the control    
    $(control).height(controlHeight+17);  
    }
    }); 

$('textarea').blur(function (e) {         
    var textLines = $(this).val().trim().split(/\r*\n/).length;      
    $(this).val($(this).val().trim()).height(textLines*17);
    });

AQUI é um post sobre isso.


6

Eu usei o plugin jQuery do Textarea Expander antes com bons resultados.


1
Uma caixa de texto (elemento de texto de entrada que eu presumo) não é multilinha. Use uma área de texto, mas estilize-a adequadamente - linhas, colunas, etc. e, em seguida, use o plug-in de crescimento automático como acima.
richsage


4
function autosize(textarea) {
    $(textarea).height(1); // temporarily shrink textarea so that scrollHeight returns content height when content does not fill textarea
    $(textarea).height($(textarea).prop("scrollHeight"));
}

$(document).ready(function () {
    $(document).on("input", "textarea", function() {
        autosize(this);
    });
    $("textarea").each(function () {
        autosize(this);
    });
});

(Isso não funcionará no Internet Explorer 9 ou anterior, pois utiliza o inputevento)


no meu projeto, como eu estou tornando as coisas inteiras na página, Jquery e datilografado, sua solução era a única maneira que eu poderia resolver o meu problema graças
Harry Sarshogh

3

Acabei de criar essa função para expandir as áreas de texto no carregamento de página. Apenas mude eachpara keyupe ocorrerá quando a área de texto for digitada.

// On page-load, auto-expand textareas to be tall enough to contain initial content
$('textarea').each(function(){
    var pad = parseInt($(this).css('padding-top'));
    if ($.browser.mozilla) 
        $(this).height(1);
    var contentHeight = this.scrollHeight;
    if (!$.browser.mozilla) 
        contentHeight -= pad * 2;
    if (contentHeight > $(this).height()) 
        $(this).height(contentHeight);
});

Testado no Chrome, IE9 e Firefox. Infelizmente, o Firefox possui esse bug que retorna o valor incorreto para scrollHeight, portanto, o código acima contém uma solução alternativa (hacky) para ele.


3

Corrigi alguns bugs na resposta fornecida por Reigel (a resposta aceita):

  1. A ordem na qual as entidades html são substituídas agora não causa código inesperado no elemento shadow. (O original substituiu ">" por "& ampgt;", causando cálculo incorreto da altura em alguns casos raros).
  2. Se o texto terminar com uma nova linha, a sombra agora terá um caractere extra "#", em vez de ter uma altura adicionada fixa, como é o caso no original.
  3. O redimensionamento da área de texto após a inicialização atualiza a largura da sombra.
  4. quebra de linha adicionada: quebra de palavra para sombra, por isso quebra o mesmo que uma área de texto (forçando quebras por palavras muito longas)

Ainda existem alguns problemas relacionados a espaços. Não vejo uma solução para espaços duplos, eles são exibidos como espaços únicos na sombra (renderização em html). Isso não pode ser resolvido usando o & nbsp ;, porque os espaços devem se romper. Além disso, a área de texto quebra uma linha após um espaço, se não houver espaço para esse espaço, ela quebrará a linha em um ponto anterior. Sugestões são bem vindas.

Código corrigido:

(function ($) {
    $.fn.autogrow = function (options) {
        var $this, minHeight, lineHeight, shadow, update;
        this.filter('textarea').each(function () {
            $this = $(this);
            minHeight = $this.height();
            lineHeight = $this.css('lineHeight');
            $this.css('overflow','hidden');
            shadow = $('<div></div>').css({
                position: 'absolute',
                'word-wrap': 'break-word',
                top: -10000,
                left: -10000,
                width: $this.width(),
                fontSize: $this.css('fontSize'),
                fontFamily: $this.css('fontFamily'),
                lineHeight: $this.css('lineHeight'),
                resize: 'none'
            }).appendTo(document.body);
            update = function () {
                shadow.css('width', $(this).width());
                var val = this.value.replace(/&/g, '&amp;')
                                    .replace(/</g, '&lt;')
                                    .replace(/>/g, '&gt;')
                                    .replace(/\n/g, '<br/>')
                                    .replace(/\s/g,'&nbsp;');
                if (val.indexOf('<br/>', val.length - 5) !== -1) { val += '#'; }
                shadow.html(val);
                $(this).css('height', Math.max(shadow.height(), minHeight));
            };
            $this.change(update).keyup(update).keydown(update);
            update.apply(this);
        });
        return this;
    };
}(jQuery));

3

Código de SpYk3HH com adição para diminuir o tamanho.

function get_height(elt) {
    return elt.scrollHeight + parseFloat($(elt).css("borderTopWidth")) + parseFloat($(elt).css("borderBottomWidth"));
}

$("textarea").keyup(function(e) {
    var found = 0;
    while (!found) {
        $(this).height($(this).height() - 10);
        while($(this).outerHeight() < get_height(this)) {
            $(this).height($(this).height() + 1);
            found = 1;
        };
    }
});

Você viu minha resposta antes da sua?
drolex

1
Sua solução não combina comigo, houve algum erro, não me lembro qual.
DSblizzard

2

Isso funcionou para mim melhor:

$('.resiText').on('keyup input', function() { 
$(this).css('height', 'auto').css('height', this.scrollHeight + (this.offsetHeight - this.clientHeight));
});
.resiText {
    box-sizing: border-box;
    resize: none;
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<textarea class="resiText"></textarea>


2

As pessoas parecem ter muito mais soluções trabalhadas ...

É assim que eu faço:

  $('textarea').keyup(function()
  {
    var 
    $this  = $(this),
    height = parseInt($this.css('line-height'),     10),
    padTop = parseInt($this.css('padding-top'),     10),
    padBot = parseInt($this.css('padding-bottom'),  10);

    $this.height(0);

    var 
    scroll = $this.prop('scrollHeight'),
    lines  = (scroll  - padTop - padBot) / height;

    $this.height(height * lines);
  });

Isso funcionará com linhas longas, além de quebras de linha. Cresce e diminui.


1

Eu escrevi essa função jquery que parece funcionar.

Você precisa especificar min-height em css e, a menos que queira codificar, ele precisa ter dois dígitos. ou seja, 12px;

$.fn.expand_ta = function() {

var val = $(this).val();
val = val.replace(/</g, "&lt;");
val = val.replace(/>/g, "&gt;");
val += "___";

var ta_class = $(this).attr("class");
var ta_width = $(this).width();

var min_height = $(this).css("min-height").substr(0, 2);
min_height = parseInt(min_height);

$("#pixel_height").remove();
$("body").append('<pre class="'+ta_class+'" id="pixel_height" style="position: absolute; white-space: pre-wrap; visibility: hidden; word-wrap: break-word; width: '+ta_width+'px; height: auto;"></pre>');
$("#pixel_height").html(val);

var height = $("#pixel_height").height();
if (val.substr(-6) == "<br />"){
    height = height + min_height;
};
if (height >= min_height) $(this).css("height", height+"px");
else $(this).css("height", min_height+"px");
}



1

Eu queria animações e encolher automaticamente. A combinação é aparentemente difícil, porque as pessoas criaram soluções bastante intensas para isso. Também a fiz à prova de múltiplas texturas. E não é tão ridiculamente pesado quanto o plugin jQuery.

Baseei-me na resposta do vsync (e na melhoria que ele fez para isso), http://codepen.io/anon/pen/vlIwj é o codepen da minha melhoria.

HTML

<textarea class='autoExpand' rows='3' data-min-rows='3' placeholder='Auto-Expanding Textarea'></textarea>

CSS

body{ background:#728EB2; }

textarea{  
  display:block;
  box-sizing: padding-box;
  overflow:hidden;

  padding:10px;
  width:250px;
  font-size:14px;
  margin:50px auto;
  border-radius:8px;
  border:6px solid #556677;
  transition:all 1s;
  -webkit-transition:all 1s;
}

JS

var rowheight = 0;

$(document).on('input.textarea', '.autoExpand', function(){
    var minRows = this.getAttribute('data-min-rows')|0,
        rows    = this.value.split("\n").length;
    $this = $(this);
    var rowz = rows < minRows ? minRows : rows;
    var rowheight = $this.attr('data-rowheight');
    if(!rowheight){
      this.rows = rowz;
      $this.attr('data-rowheight', (this.clientHeight  - parseInt($this.css('padding-top')) - parseInt($this.css('padding-bottom')))/ rowz);
    }else{
      rowz++;
      this.style.cssText = 'height:' + rowz * rowheight + 'px'; 
    }
});

Nota: notei que às vezes funciona melhor com o tamanho da caixa: caixa de conteúdo. Eu não sou inteiramente certo porque, porém, ele deve processar o preenchimento corretamente :(
Lodewijk

1

Há muitas respostas para isso, mas achei algo muito simples, anexe um evento de keyup à área de texto e verifique se a tecla Enter é pressionada, o código da tecla é 13

keyPressHandler(e){ if(e.keyCode == 13){ e.target.rows = e.target.rows + 1; } }

Isso adicionará outra linha à sua área de texto e você poderá estilizar a largura usando CSS.


1

Digamos que você esteja tentando fazer isso usando o Knockout ... veja como:

Na página:

<textarea data-bind="event: { keyup: $root.GrowTextArea }"></textarea>

No modelo de exibição:

self.GrowTextArea = function (data, event) {
    $('#' + event.target.id).height(0).height(event.target.scrollHeight);
}

Isso deve funcionar mesmo se você tiver várias áreas de texto criadas por um forock de Knockout, como eu.


1

Solução Simples:

HTML:

<textarea class='expand'></textarea>

JS:

$('textarea.expand').on('input', function() {
  $(this).scrollTop($(this).height());
});
$('textarea.expand').scroll(function() {
  var h = $(this).scrollTop();
  if (h > 0)
    $(this).height($(this).height() + h);
});

https://fiddle.jshell.net/7wsnwbzg/


1

A solução mais simples:

html:

<textarea class="auto-expand"></textarea>

css:

.auto-expand {
    overflow:hidden;
    min-height: 80px;
}

js (jquery):

$(document).ready(function () {
 $("textarea.auto-expand").focus(function () {
        var $minHeight = $(this).css('min-height');
        $(this).on('input', function (e) {
            $(this).css('height', $minHeight);
            var $newHeight = $(this)[0].scrollHeight;
            $(this).css('height', $newHeight);
        });
    });       
});

1

Solução com JS puro

function autoSize() {
  if (element) {
    element.setAttribute('rows', 2) // minimum rows
    const rowsRequired = parseInt(
      (element.scrollHeight - TEXTAREA_CONFIG.PADDING) / TEXTAREA_CONFIG.LINE_HEIGHT
    )
    if (rowsRequired !== parseInt(element.getAttribute('rows'))) {
      element.setAttribute('rows', rowsRequired)
    }
  }
}

https://jsfiddle.net/Samb102/cjqa2kf4/54/


1

Esta é a solução que acabei usando. Eu queria uma solução em linha, e isso até agora parece funcionar muito bem:

<textarea onkeyup="$(this).css('height', 'auto').css('height', this.scrollHeight + this.offsetHeight - this.clientHeight);"></textarea>

1

function autoResizeTextarea() {
  for (let index = 0; index < $('textarea').length; index++) {
    let element = $('textarea')[index];
    let offset = element.offsetHeight - element.clientHeight;
    $(element).css('resize', 'none');
    $(element).on('input', function() {
      $(this).height(0).height(this.scrollHeight - offset - parseInt($(this).css('padding-top')));
    });
  }
}

https://codepen.io/nanachi1/pen/rNNKrzQ

isso deve funcionar.


0

@Georgiy Ivankin fez uma sugestão em um comentário, usei-a com sucesso :) -, mas com pequenas alterações:

$('#note').on('keyup',function(e){
    var maxHeight = 200; 
    var f = document.getElementById('note'); 
    if (f.clientHeight < f.scrollHeight && f.scrollHeight < maxHeight ) 
        { f.style.height = f.scrollHeight + 'px'; }
    });      

Ele para de se expandir após atingir a altura máxima de 200px


0

Pergunta antiga, mas você poderia fazer algo assim:

html:

<textarea class="text-area" rows="1"></textarea>

jquery:

var baseH; // base scroll height

$('body')
    .one('focus.textarea', '.text-area', function(e) {
        baseH = this.scrollHeight;
    })
    .on('input.textarea', '.text-area', function(e) {
        if(baseH < this.scrollHeight) {
            $(this).height(0).height(this.scrollHeight);
        }
        else {
            $(this).height(0).height(baseH);
        }
    });

Dessa forma, o redimensionamento automático será aplicado a qualquer área de texto com a classe "área de texto". Também diminui quando o texto é removido.

jsfiddle:

https://jsfiddle.net/rotaercz/46rhcqyn/


0

Tente o seguinte:

  $('textarea[name="mytextarea"]').on('input', function(){
    $(this).height('auto').height($(this).prop('scrollHeight') + 'px');
  }).trigger('input');

0

Solução jQuery simples:

$("textarea").keyup(function() {
    var scrollHeight = $(this).prop('scrollHeight') - parseInt($(this).css("paddingTop")) - parseInt($(this).css("paddingBottom"));

    if (scrollHeight > $(this).height()) {
        $(this).height(scrollHeight + "px");
    }
});

HTML:

<textarea rows="2" style="padding: 20px; overflow: hidden; resize: none;"></textarea>

O estouro deve estar oculto . Redimensionar é nenhum se você não deseja redimensioná-lo com o mouse.

Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.