Já existem muitas respostas boas aqui. Por diversão, implementei a solução abaixo, com base nas outras respostas e nas minhas próprias ideias.
<input class="adjust">
O elemento de entrada é ajustado com precisão de pixel e um deslocamento adicional pode ser definido.
function adjust(elements, offset, min, max) {
offset = offset || 0;
min = min || 0;
max = max || Infinity;
elements.each(function() {
var element = $(this);
var id = btoa(Math.floor(Math.random() * Math.pow(2, 64)));
var tag = $('<span id="' + id + '">' + element.val() + '</span>').css({
'display': 'none',
'font-family': element.css('font-family'),
'font-size': element.css('font-size'),
}).appendTo('body');
function update() {
setTimeout(function() {
tag.html(element.val().replace(/ /g, ' '));
var size = Math.max(min, Math.min(max, tag.width() + offset));
if (size < max)
element.scrollLeft(0);
element.width(size);
}, 0);
};
update();
element.keydown(update);
});
}
adjust($('.adjust'), 10, 100, 500);
O ajuste é suavizado com uma transição CSS.
.adjust {
transition: width .15s;
}
Aqui está o violino . Espero que isso possa ajudar outras pessoas que procuram uma solução limpa.